Author: James Mounsey-Moran

  • Add Animated Icons with Conditional Formatting

    Add Animated Icons with Conditional Formatting

    I use icons quite regularly as part of conditional formatting whether it’s to show change or highlight certain values. Working with Prism, some of those icons were custom-made. But what I didn’t realise until recently is that they don’t have to be static images.

    You can apply the same SVG animation techniques I’ve been experimenting with elsewhere, convert them to base64, and embed them directly into a Power BI theme!

    So, keep reading to learn how to add animated icons using Power BI’s conditional formatting.

    I began with a couple of static icons I’d previously built, converted them to SVG, and added a simple transform animation.

    Here’s an example SVG: a chevron that switches colours and moves left to right in a loop.

    <svg width="100" height="100" viewBox="0 0 100 100" xmlns="http://www.w3.org/2000/svg">
      <polyline points="40,30 60,50 40,70" fill="none" stroke="#ffffff" stroke-width="8" stroke-linecap="round" stroke-linejoin="round">
        <animateTransform 
          attributeName="transform" 
          type="translate" 
          values="-8 0;8 0;-8 0" 
          dur="2s" 
          repeatCount="indefinite" />
        <animate 
          attributeName="stroke" 
          values="#ffffff;#6bfad8;#ffffff" 
          dur="2s" 
          repeatCount="indefinite" />
      </polyline>
    </svg>
    


    I recommend using svgviewer.dev a brilliant tool I’ve mentioned in previous posts.

    • Paste your SVG code into the viewer to preview the animation.
    • Switch to the Data URI tab.
    • Copy the base64 output and it’s ready to use!

    Here’s the base64 version of the chevron above (copied here for convenience):

    data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMTAwIiBoZWlnaHQ9IjEwMCIgdmlld0JveD0iMCAwIDEwMCAxMDAiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CiAgPHBvbHlsaW5lIHBvaW50cz0iNDAsMzAgNjAsNTAgNDAsNzAiIGZpbGw9Im5vbmUiIHN0cm9rZT0iI2ZmZmZmZiIgc3Ryb2tlLXdpZHRoPSI4IiBzdHJva2UtbGluZWNhcD0icm91bmQiIHN0cm9rZS1saW5lam9pbj0icm91bmQiPgogICAgPGFuaW1hdGVUcmFuc2Zvcm0gCiAgICAgIGF0dHJpYnV0ZU5hbWU9InRyYW5zZm9ybSIgCiAgICAgIHR5cGU9InRyYW5zbGF0ZSIgCiAgICAgIHZhbHVlcz0iLTggMDs4IDA7LTggMCIgCiAgICAgIGR1cj0iMnMiIAogICAgICByZXBlYXRDb3VudD0iaW5kZWZpbml0ZSIgLz4KICAgIDxhbmltYXRlIAogICAgICBhdHRyaWJ1dGVOYW1lPSJzdHJva2UiIAogICAgICB2YWx1ZXM9IiNmZmZmZmY7IzZiZmFkODsjZmZmZmZmIiAKICAgICAgZHVyPSIycyIgCiAgICAgIHJlcGVhdENvdW50PSJpbmRlZmluaXRlIiAvPgogIDwvcG9seWxpbmU+Cjwvc3ZnPgo=

    Now let’s embed this into a Power BI theme. If you’re already using a custom theme, you can add to it, or create a new one. Here’s a basic example with several icons:

    {
      "name": "SVGTHEME",
      "icons": {
        "GreenChev": {
          "description": "GreenChev",
          "url": "data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMTAwIiBoZWlnaHQ9IjEwMCIgdmlld0JveD0iMCAwIDEwMCAxMDAiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CiAgPHBvbHlsaW5lIHBvaW50cz0iNDAsMzAgNjAsNTAgNDAsNzAiIGZpbGw9Im5vbmUiIHN0cm9rZT0iI2ZmZmZmZiIgc3Ryb2tlLXdpZHRoPSI4IiBzdHJva2UtbGluZWNhcD0icm91bmQiIHN0cm9rZS1saW5lam9pbj0icm91bmQiPgogICAgPGFuaW1hdGVUcmFuc2Zvcm0gYXR0cmlidXRlTmFtZT0idHJhbnNmb3JtIiB0eXBlPSJ0cmFuc2xhdGUiIHZhbHVlcz0iLTggMDs4IDA7LTggMCIgZHVyPSIycyIgcmVwZWF0Q291bnQ9ImluZGVmaW5pdGUiIC8+CiAgICA8YW5pbWF0ZSBhdHRyaWJ1dGVOYW1lPSJzdHJva2UiIHZhbHVlcz0iI2ZmZmZmZjsjNmJmYWQ4OyNmZmZmZmYiIGR1cj0iMnMiIHJlcGVhdENvdW50PSJpbmRlZmluaXRlIiAvPgogIDwvcG9seWxpbmU+Cjwvc3ZnPg=="
        },
        "PurpChev": {
          "description": "PurpChev",
          "url": "data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMTAwIiBoZWlnaHQ9IjEwMCIgdmlld0JveD0iMCAwIDEwMCAxMDAiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CiAgPHBvbHlsaW5lIHBvaW50cz0iNDAsMzAgNjAsNTAgNDAsNzAiIGZpbGw9Im5vbmUiIHN0cm9rZT0iI2ZmZmZmZiIgc3Ryb2tlLXdpZHRoPSI4IiBzdHJva2UtbGluZWNhcD0icm91bmQiIHN0cm9rZS1saW5lam9pbj0icm91bmQiPgogICAgPGFuaW1hdGVUcmFuc2Zvcm0gYXR0cmlidXRlTmFtZT0idHJhbnNmb3JtIiB0eXBlPSJ0cmFuc2xhdGUiIHZhbHVlcz0iLTggMDs4IDA7LTggMCIgZHVyPSIycyIgcmVwZWF0Q291bnQ9ImluZGVmaW5pdGUiIC8+CiAgICA8YW5pbWF0ZSBhdHRyaWJ1dGVOYW1lPSJzdHJva2UiIHZhbHVlcz0iI2ZmZmZmZjsjOWI1OWI2OyNmZmZmZmYiIGR1cj0iMnMiIHJlcGVhdENvdW50PSJpbmRlZmluaXRlIiAvPgogIDwvcG9seWxpbmU+Cjwvc3ZnPg=="
        },
        "Warn": {
          "description": "Warn",
          "url": "data:image/svg+xml;base64,PHN2ZyB2aWV3Qm94PSIwIDAgMTAwIDEwMCIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICAgIDxzdHlsZT4KICAgICAgICAvKiBEZWZpbmUgdGhlIGFuaW1hdGlvbiBmb3IgdGhlIGV4Y2xhbWF0aW9uIG1hcmsgd2l0aGluIFNWRyBzdHlsZSAqLwogICAgICAgIC8qIE1vZGlmaWVkIGZvciBhIHNtb290aGVyIHB1bHNlIGVmZmVjdCAqLwogICAgICAgIEBrZXlmcmFtZXMgc3ZnUHVsc2UgewogICAgICAgICAgICAwJSB7CiAgICAgICAgICAgICAgICB0cmFuc2Zvcm06IHNjYWxlKDEpOwogICAgICAgICAgICAgICAgb3BhY2l0eTogMTsKICAgICAgICAgICAgfQogICAgICAgICAgICA1MCUgewogICAgICAgICAgICAgICAgdHJhbnNmb3JtOiBzY2FsZSgwLjkpOyAvKiBTbGlnaHRseSBsYXJnZXIgZm9yIGEgc3VidGxlIHB1bHNlICovCiAgICAgICAgICAgICAgICBvcGFjaXR5OiAwLjk7IC8qIFNsaWdodCBmYWRlIGZvciBhIHNvZnRlciBwdWxzZSAqLwogICAgICAgICAgICB9CiAgICAgICAgICAgIDEwMCUgewogICAgICAgICAgICAgICAgdHJhbnNmb3JtOiBzY2FsZSgxKTsKICAgICAgICAgICAgICAgIG9wYWNpdHk6IDE7CiAgICAgICAgICAgIH0KICAgICAgICB9CgogICAgICAgIC5zdmctZXhjbGFtYXRpb24tcGF0aCB7CiAgICAgICAgICAgIGZpbGw6ICNGQUYwQ0E7IC8qIFJlZCBjb2xvciBmb3IgdGhlIGV4Y2xhbWF0aW9uIG1hcmsgKi8KICAgICAgICAgICAgdHJhbnNmb3JtLW9yaWdpbjogNTBweCA1MHB4OyAvKiBDZW50ZXIgb2YgdGhlIHZpZXdCb3ggZm9yIHNjYWxpbmcgKi8KICAgICAgICAgICAgYW5pbWF0aW9uOiBzdmdQdWxzZSAxLjVzIGVhc2UtaW4tb3V0IGluZmluaXRlOyAvKiBBcHBseSB0aGUgbmV3IHB1bHNlIGFuaW1hdGlvbiAqLwogICAgICAgIH0KICAgIDwvc3R5bGU+CiAgICA8IS0tIEV4Y2xhbWF0aW9uIG1hcmsgYm9keSAtLT4KICAgIDxyZWN0IHg9IjQ1IiB5PSIxMCIgd2lkdGg9IjEwIiBoZWlnaHQ9IjYwIiByeD0iNSIgcnk9IjUiIGNsYXNzPSJzdmctZXhjbGFtYXRpb24tcGF0aCIgLz4KICAgIDwhLS0gRXhjbGFtYXRpb24gbWFyayBkb3QgLS0+CiAgICA8Y2lyY2xlIGN4PSI1MCIgY3k9Ijg1IiByPSI3IiBjbGFzcz0ic3ZnLWV4Y2xhbWF0aW9uLXBhdGgiIC8+Cjwvc3ZnPgo="
        },
        "Error": {
          "description": "Error",
          "url": "data:image/svg+xml;base64,PHN2ZyB2aWV3Qm94PSIwIDAgMTAwIDEwMCIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIj4KICAgIDxzdHlsZT4KICAgICAgICAvKiBEZWZpbmUgdGhlIGFuaW1hdGlvbiBmb3IgdGhlIGNyb3NzIG1hcmsgKi8KICAgICAgICBAa2V5ZnJhbWVzIHN2Z1NoYWtlIHsKICAgICAgICAgICAgMCUgewogICAgICAgICAgICAgICAgdHJhbnNmb3JtOiByb3RhdGUoMGRlZyk7CiAgICAgICAgICAgIH0KICAgICAgICAgICAgMTAlIHsKICAgICAgICAgICAgICAgIHRyYW5zZm9ybTogcm90YXRlKC01ZGVnKTsgLyogUXVpY2sgcm90YXRpb24gdG8gdGhlIGxlZnQgKi8KICAgICAgICAgICAgfQogICAgICAgICAgICAyMCUgewogICAgICAgICAgICAgICAgdHJhbnNmb3JtOiByb3RhdGUoNWRlZyk7IC8qIFF1aWNrIHJvdGF0aW9uIHRvIHRoZSByaWdodCAqLwogICAgICAgICAgICB9CiAgICAgICAgICAgIDMwJSB7CiAgICAgICAgICAgICAgICB0cmFuc2Zvcm06IHJvdGF0ZSgwZGVnKTsgLyogUXVpY2sgcmV0dXJuIHRvIGNlbnRlciAqLwogICAgICAgICAgICB9CiAgICAgICAgICAgIC8qIEZyb20gMzAlIHRvIDEwMCUsIHRoZSBjcm9zcyByZW1haW5zIGF0IDAgZGVncmVlcywgY3JlYXRpbmcgYSBwYXVzZSAqLwogICAgICAgICAgICAxMDAlIHsKICAgICAgICAgICAgICAgIHRyYW5zZm9ybTogcm90YXRlKDBkZWcpOwogICAgICAgICAgICB9CiAgICAgICAgfQoKICAgICAgICAuc3ZnLWNyb3NzLXBhdGggewogICAgICAgICAgICBmaWxsOiAjZGMyNjI2OyAvKiBSZWQgY29sb3IgZm9yIHRoZSBjcm9zcyBtYXJrICovCiAgICAgICAgICAgIHRyYW5zZm9ybS1vcmlnaW46IDUwcHggNTBweDsgLyogQ2VudGVyIG9mIHRoZSB2aWV3Qm94IGZvciByb3RhdGlvbiAqLwogICAgICAgICAgICBhbmltYXRpb246IHN2Z1NoYWtlIDJzIGVhc2UtaW4tb3V0IGluZmluaXRlOyAvKiBBcHBseSB0aGUgc2hha2UgYW5pbWF0aW9uLCB3aXRoIGEgbG9uZ2VyIGR1cmF0aW9uIHRvIGFjY29tbW9kYXRlIHRoZSBwYXVzZSAqLwogICAgICAgIH0KICAgIDwvc3R5bGU+CiAgICA8IS0tIEdyb3VwIHRvIGFwcGx5IG92ZXJhbGwgcm90YXRpb24gdG8gdGhlIGNyb3NzIC0tPgogICAgPGcgdHJhbnNmb3JtPSJyb3RhdGUoNDUgNTAgNTApIj4KICAgICAgICA8IS0tIEZpcnN0IGxpbmUgb2YgdGhlIGNyb3NzIChpbml0aWFsbHkgdmVydGljYWwsIHJvdGF0ZWQgKzQ1IGRlZ3JlZXMpIC0tPgogICAgICAgIDxyZWN0IHg9IjQ1IiB5PSIxNSIgd2lkdGg9IjEwIiBoZWlnaHQ9IjcwIiByeD0iNSIgcnk9IjUiIHRyYW5zZm9ybT0icm90YXRlKDQ1IDUwIDUwKSIgY2xhc3M9InN2Zy1jcm9zcy1wYXRoIiAvPgogICAgICAgIDwhLS0gU2Vjb25kIGxpbmUgb2YgdGhlIGNyb3NzIChpbml0aWFsbHkgaG9yaXpvbnRhbCwgcm90YXRlZCAtNDUgZGVncmVlcykgLS0+CiAgICAgICAgPHJlY3QgeD0iMTUiIHk9IjQ1IiB3aWR0aD0iNzAiIGhlaWdodD0iMTAiIHJ4PSI1IiByeT0iNSIgdHJhbnNmb3JtPSJyb3RhdGUoLTQ1IDUwIDUwKSIgY2xhc3M9InN2Zy1jcm9zcy1wYXRoIiAvPgogICAgPC9nPgo8L3N2Zz4K"
        }
      }
    }
    

    Save this as a .json file, for example: svgicons.json.

    Each icon entry includes the below:

    • Name (What you want to refer to the icon as)
    • Description (A brief description of the Icon)
    • URL (The base64 of the icon)

    If you want to add additional icons, just load them in with the same format.

    Once you have the theme ready, we need to load it into PowerBI

    • Open Power BI Desktop.
    • Go to the View tab → click the Themes dropdown → select Browse for themes.
    • Upload your .json file.

    Once the file is loaded in we have the icons ready to be used! So lets try it!

    • Select a column in your table or matrix.
    • Go to Conditional formatting → Icons.
    • Add your desired rules.
    • Scroll to the bottom of the icon list you’ll see your custom ones added from the theme.
      • Note: Previews won’t animate or show properly in the selection list so make sure your naming convention is clear!

    Once added in you should end up with something like the below:

    Here we have

    • Chevrons > (green and purple)
    • Warning ! with a pulse effect
    • Error X with a slight wobble effect

    I love finding new ways to enhance what Power BI can do natively especially when it unlocks more meaningful visualisations.

    Got questions or want to explore more ideas like this? Feel free to get in touch I’m always happy to help!

    Thanks for reading!!

  • Creating Card Slicers in PowerBI

    Creating Card Slicers in PowerBI

    One of the things I hear the most when running demos or working with customers.

    “If I click on that KPI will that then filter the report”

    Sadly, no. Or well not by default. Kick in Product Manager brain, make it work!

    So in this blog post I will be running through my implementation of creating cards that can do exactly that, be clicked on, filter the rest of the report and fit nicely with the overall aesthetic I’ve built with Prism. We can also throw in some SVG animation to enhance everything visually and give it that extra level of polish.

    So, how do we make KPI cards clickable?

    Power BI doesn’t natively allow KPIs to act as slicers, so we need a workaround.

    To be able to click and filter in PowerBI we have a few slicer options, but to make it look like a KPI card that’s when we need to consider some options:

    We could:

    • Layer it (create a card and then add a dummy button slicer on top)
    • Use a Button Slicer

    In this example, I’ve gone with the button slicer approach. It integrates cleanly into the overall product UI and is easier to manage as I evolve the design further.

    The trick: SVG + Measures = Interactive KPI

    In some previous posts, Ive used HTML / SVG with measures that we can then add to the image url field. Ive used the same logic here but then also added a state for Static, Hover and Selected. As we also need to think about making it obvious for end users to see we’ve added a filter.

    Some subtle icons and animation can be really effective here

    Tutorial

    To start off:

    • Create a button slicer
    • Add the 3 different measures below

    Below are my SVG measures. Each one displays a simple KPI-style card with a filter icon and adjusts styling such as colour and animation based on state (Static, Hover, Selected). The end result looks something like this:

    HTML_Card_Filter = 
    
    VAR pct = COUNTROWS(
        FILTER(
            Sheet1,
            Sheet1[allocationfilter] = 1
        )
    )
    VAR sizeWidth = 300
    VAR sizeHeight = 110
    
    -- Colors
    VAR baseColor = "#373737"
    VAR textColor = "#f5f5f5"
    
    -- SVG generation
    VAR svg = "
    <svg xmlns='http://www.w3.org/2000/svg' width='" & sizeWidth & "' height='" & sizeHeight & "' viewBox='0 0 " & sizeWidth & " " & sizeHeight & "'>
      <style>
        .card-base {
          fill: " & baseColor & ";
        }
        .accent {
          fill: #6BFAD8;
        }
        .title {
          font-family: Segoe UI, sans-serif;
          font-size: 14px;
          font-weight: 600;
          fill: " & textColor & ";
          text-anchor: end;
        }
        .value {
          font-family: Segoe UI, sans-serif;
          font-size: 42px;
          font-weight: 600;
          fill: " & textColor & ";
          text-anchor: end;
        }
      </style>
    
      <!-- Background -->
      <rect class='card-base' width='" & sizeWidth & "' height='" & sizeHeight & "' rx='10' ry='10'/>
      
      <!-- Right accent -->
      <path class='accent' d='M " & (sizeWidth - 5) & " 0 L " & (sizeWidth - 5) & " " & sizeHeight & " Q " & sizeWidth & " " & sizeHeight & " " & sizeWidth & " " & (sizeHeight - 10) & " L " & sizeWidth & " 10 Q " & sizeWidth & " 0 " & (sizeWidth - 5) & " 0 Z'/>
      
      <!-- Filter icon in top left -->
      <g transform='translate(18, 18)'>
        <path d='M2 3h12l-4 4v4l-4 2V7L2 3z' stroke='" & textColor & "' stroke-width='1.5' fill='none' stroke-linecap='round' stroke-linejoin='round'/>
      </g>
    
      <!-- Text -->
      <text x='" & (sizeWidth - 30) & "' y='60' class='value'>" & pct & "</text>
      <text x='" & (sizeWidth - 30) & "' y='80' class='title'>&lt;50% Allocated</text>
    </svg>
    "
    
    RETURN "data:image/svg+xml;utf8," & svg
    
    HTML_Card_Filter_Hover = 
    
    VAR pct = COUNTROWS(
        FILTER(
            Sheet1,
            Sheet1[allocationfilter] = 1
        )
    )
    VAR sizeWidth = 300
    VAR sizeHeight = 110
    VAR animationKey = CONCATENATE("anim-", FORMAT(NOW(), "hhmmss"))
    
    -- Colors
    VAR baseColor = "#373737"
    VAR textColor = "#f5f5f5"
    
    -- SVG generation
    VAR svg = "
    <svg xmlns='http://www.w3.org/2000/svg' width='" & sizeWidth & "' height='" & sizeHeight & "' viewBox='0 0 " & sizeWidth & " " & sizeHeight & "'>
      <style>
        .card-base {
          fill: " & baseColor & ";
        }
        .accent {
          fill: #6BFAD8;
        }
        .title {
          font-family: Segoe UI, sans-serif;
          font-size: 14px;
          font-weight: 600;
          fill: " & textColor & ";
          text-anchor: end;
        }
        .value {
          font-family: Segoe UI, sans-serif;
          font-size: 42px;
          font-weight: 600;
          fill: " & textColor & ";
          text-anchor: end;
        }
        .animated-rect {
          fill: #6BFAD8;
          opacity: 0.7;
        }
      </style>
    
      <!-- Background -->
      <rect class='card-base' width='" & sizeWidth & "' height='" & sizeHeight & "' rx='10' ry='10'/>
      
      <!-- Right accent -->
      <path class='accent' d='M " & (sizeWidth - 5) & " 0 L " & (sizeWidth - 5) & " " & sizeHeight & " Q " & sizeWidth & " " & sizeHeight & " " & sizeWidth & " " & (sizeHeight - 10) & " L " & sizeWidth & " 10 Q " & sizeWidth & " 0 " & (sizeWidth - 5) & " 0 Z'/>
      
      <!-- Filter icon in top left -->
      <g transform='translate(18, 18)'>
        <!-- Static rectangle over filter icon -->
        <rect x='2' y='16' width='12' height='4' rx='2' class='animated-rect' data-anim='" & animationKey & "'/>
        <path d='M2 3h12l-4 4v4l-4 2V7L2 3z' stroke='" & textColor & "' stroke-width='1.5' fill='none' stroke-linecap='round' stroke-linejoin='round'/>
      </g>
    
      <!-- Text -->
      <text x='" & (sizeWidth - 30) & "' y='60' class='value'>" & pct & "</text>
    <text x='" & (sizeWidth - 30) & "' y='80' class='title'>&lt;50% Allocated</text>
    </svg>
    "
    
    RETURN "data:image/svg+xml;utf8," & svg
    
    
    HTML_Card_Filter_Selected = 
    
    VAR pct = COUNTROWS(
        FILTER(
            Sheet1,
            Sheet1[allocationfilter] = 1
        )
    )
    VAR sizeWidth = 300
    VAR sizeHeight = 110
    VAR animationKey = CONCATENATE("anim-", FORMAT(NOW(), "hhmmss"))
    
    -- Colors
    VAR baseColor = "#373737"
    VAR textColor = "#f5f5f5"
    
    -- SVG generation
    VAR svg = "
    <svg xmlns='http://www.w3.org/2000/svg' width='" & sizeWidth & "' height='" & sizeHeight & "' viewBox='0 0 " & sizeWidth & " " & sizeHeight & "'>
      <style>
        .card-base {
          fill: " & baseColor & ";
        }
        .accent {
          fill: #6BFAD8;
        }
        .title {
          font-family: Segoe UI, sans-serif;
          font-size: 14px;
          font-weight: 600;
          fill: " & textColor & ";
          text-anchor: end;
        }
        .value {
          font-family: Segoe UI, sans-serif;
          font-size: 42px;
          font-weight: 600;
          fill: " & textColor & ";
          text-anchor: end;
        }
        .animated-rect {
          fill: #6BFAD8;
          opacity: 0.9;
        }
        .pulse-glow {
          fill: none;
          stroke: rgba(255, 255, 255, 0.8);
          stroke-width: 1;
          opacity: 0;
        }
      </style>
    
      <!-- Background -->
      <rect class='card-base' width='" & sizeWidth & "' height='" & sizeHeight & "' rx='10' ry='10'/>
      <!-- Right accent -->
      <path class='accent' d='M " & (sizeWidth - 5) & " 0 L " & (sizeWidth - 5) & " " & sizeHeight & " Q " & sizeWidth & " " & sizeHeight & " " & sizeWidth & " " & (sizeHeight - 10) & " L " & sizeWidth & " 10 Q " & sizeWidth & " 0 " & (sizeWidth - 5) & " 0 Z'/>
      
      
      <!-- Filter icon in top left -->
      <g transform='translate(18, 18)'>
        <!-- Pulsing rectangle over filter icon -->
        <rect x='2' y='16' width='12' height='4' rx='2' class='animated-rect'>
          <animate attributeName='opacity' values='0.9;0.1;0.9' dur='1.5s' repeatCount='indefinite' begin='0s'/>
          <animate attributeName='fill' values='#6BFAD8;#4ECDC4;#6BFAD8' dur='1.5s' repeatCount='indefinite' begin='0s'/>
        </rect>
        <path d='M2 3h12l-4 4v4l-4 2V7L2 3z' stroke='#6BFAD8' stroke-width='1.5' fill='none' stroke-linecap='round' stroke-linejoin='round'/>
      </g>
    
      <!-- Text -->
      <text x='" & (sizeWidth - 30) & "' y='60' class='value'>" & pct & "</text>
    <text x='" & (sizeWidth - 30) & "' y='80' class='title'>&lt;50% Allocated</text>
    </svg>
    "
    
    RETURN "data:image/svg+xml;utf8," & svg
    

    Once you have added your button slicer and added the above measures, we need to apply them to the different states and ensure all looks and works as required.

    In my example, I used a calculated column linked to my table that was simply filtered on a 1 or 0 depending on the conditions – then linked to the wider data. (The data I used was looking at allocation rates, over 50% was a 1 everything else a 0)

    Next:

    • On the filters if required set the filter on that visual to show what you are looking to filter on (i.e “1”)
    • Under layout, we need to show only the 1 row and column and I generally set the space between buttons as 0

    Now to prepare for using the image measures we need to hide everything else

    Under General

    • Disable Title
    • Disable Header Icons
    • Disable Effects / Background

    Under Visual / Buttons

    • Disable Border
    • Disable Full
    • Padding – Set to custom and set all to 0 px

    Under Callout Values

    • Disable Values

    You should now essentially have a box that shows nothing!

    Now back under visual / Image we need to apply the 3 measures – apply the following based on the state

    AllHoverSelected
    HTML_Card_FilterHTML_Card_HoverHTML_Card_Selected

    With all those added you should now have you Card Slicer!

    Now you will need to use fields and titles that make sense to your data, those can all be changed within the measures themselves, just dont forget to change in all 3!

    Once complete you should end up with something like the following:

    Why This Matters

    Beyond just looking nice, this pattern supports:

    • Design flexibility: Themed, responsive KPIs tailored to your report’s look and feel
    • Better UX: Users get clear, visual feedback
    • Faster interactions: No more guessing what’s clickable

    Thanks for reading! As always I would love to hear from you, as well as any ideas you have!

  • Creating SVG Powered Navigation Buttons in PowerBI

    Creating SVG Powered Navigation Buttons in PowerBI

    Not every SVG needs to be used as a measure in Power BI. In some cases, it makes more sense to take a simpler approach and use .SVG files directly. This is especially helpful when working with buttons in Power BI, which currently do not support imageURL (like we would use in the new card visuals, for example) and instead only allow image input. Hopefully this blog post helps you with creating SVG powered navigation buttons in PowerBI.

    Thankfully, those image options give us all we need to make buttons more dynamic. You can use animated SVG icons or even animate the entire button if needed.

    Power BI buttons let you define multiple visual states, including Default, Hover, Selected etc. Each state can be customised with different settings for Text, Icon, Fill, and Border. Both Icon and Fill accept image files.

    SVG files can contain coded information, which means you can embed animation directly into them. You just need to make sure the .SVG file is correctly structured and works as expected inside Power BI.

    A useful tool for testing and previewing SVG files is https://www.svgviewer.dev.

    This example uses the Icon section of the button visual. Here, we apply an animated SVG that responds to the button state, such as when a user hovers over it.

    For this example, I am creating a simple arrow that moves up and down when the button is hovered over. We need two versions of the SVG. One will be static for the default state. The other will contain the animation for the hover state.

    Note: Make sure to include the namespace in your SVG. This is required for it to render correctly in Power BI.

    <?xml version="1.0" encoding="UTF-8" standalone="no"?>
    <svg 
       xmlns="http://www.w3.org/2000/svg"
    width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="1" stroke-linecap="round" stroke-linejoin="round">
      <line x1="6" y1="18" x2="18" y2="6" />
      <polyline points="10 6 18 6 18 14" />
    </svg>
    <?xml version="1.0" encoding="UTF-8" standalone="no"?>
    <svg 
       xmlns="http://www.w3.org/2000/svg"
    width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="white" stroke-width="1" stroke-linecap="round" stroke-linejoin="round">
      <g>
        <animateTransform
          attributeName="transform"
          type="translate"
          values="0 0; 1 -1; 2 -2; 3 -3; 4 -4; 0 0"
          keyTimes="0; 0.1; 0.25; 0.45; 0.8; 1"
          dur="2s"
          repeatCount="indefinite"
        />
        <line x1="6" y1="18" x2="18" y2="6" />
        <polyline points="10 6 18 6 18 14" />
      </g>
    </svg>

    Copy the code up to https://www.svgviewer.dev/ to preview, from here you can also download the .svg file as well which makes things nice and simple!

    Into PowerBI, Insert a new blank button, select the button to format at and under Style / Icon, choose Icon Type – Custom

    Next we need to apply the two image files, one for the default status and the other for hover

    Default

    • Under Style Apply Settings to: Firstly make the state as Default
    • Next, go back to the Icon section and apply the static SVG. I set the image fit as normal
    • Tweak the placement as required using alignment and padding ( I used left horizontal alignment, middle vertical, with padding of 10 on the left and 7 at the bottom.
    • I then set the icon size at 40, but against this can be tweaked as required

    Hover

    • This time under Style Apply Settings to: Switch to Hover
    • Back Under Icon, apply the animated svg. And set the image fit the same as your default setting (especially if using a similar icon for each)
    • If using the same icon like I have in this example ensure your placement and size matches the default settings
    • In my example I also added an extra border around the button, feel free to experiment and see what you like best!

    This example instead uses the Fill option of the button visual. Again, we apply an animated SVG that responds to the button state, such as when a user hovers over it but this time instead of affecting an icon, we can affect the button itself.

    For this example, a wave animation down the left hand side of the button that appears once hovered over. Unlike the icon example, we only need one SVG file this time as our default state for this example is just a solid fill background (just make sure your background matches the colour we are adding for the svg)

    Note: Make sure to include the namespace in your SVG. This is required for it to render correctly in Power BI.

    <svg xmlns="http://www.w3.org/2000/svg" width="259" height="74" viewBox="0 0 259 74">
      <style>
        .wave-accent {
          fill: #6bfad8;
          animation: waveMove 4s linear infinite;
        }
    
        @keyframes waveMove {
          0% { transform: translateY(0); }
          100% { transform: translateY(-160px); }
        }
      </style>
      <rect x="0" y="0" width="259" height="74" rx="6" ry="6" fill="#292929" />
      <defs>
        <clipPath id="waveClip">
          <rect x="0" y="0" width="25" height="74" />
        </clipPath>
      </defs>
      <g clip-path="url(#waveClip)">
        <g class="wave-accent">
          <path d="M12.5,0 q10,10 0,20 -10,10 0,20 10,10 0,20 -10,10 0,20 10,10 0,20 -10,10 0,20 10,10 0,20 -10,10 0,20 H0 V0 Z" />
          <path d="M12.5,160 q10,10 0,20 -10,10 0,20 10,10 0,20 -10,10 0,20 10,10 0,20 -10,10 0,20 10,10 0,20 -10,10 0,20 H0 V160 Z" />
        </g>
      </g>
    </svg>
    

    Copy the code up to https://www.svgviewer.dev/ to preview and download the .SVG file.

    Note this time, we have added the width and height in various parts of the code. This needs to match up with your size of the button in PowerBI to ensure if fills the whole area

    Into PowerBI, Insert a new blank button, select the button to format at and under Style / Icon, choose Icon Type – Custom

    Next we need to apply the single image file for the hover status

    Default

    • Under Style Apply Settings to: Ensure we are set on Default
    • Ensure the fill colour is set to exactly the same colour being used within your button
    • You may want to tweak the positioning of the text on your button, I applied middle horizontal and vertical alignment and then moved it around with padding

    Hover

    • Under Style Apply Settings to: Switch to Hover
    • Under Fill, browse and apply the animated svg. In this instance I’ve set the Image Fit just as Normal

    And that now should be it! You should have two different types of animated navigation buttons, you can apply the actions you need them to do down in the button settings and you are sorted!

    Please play around experiment with different types and shapes and of course share!

    Thanks for reading!