Resolved What am I doing wrong with StateHasChanged

JohnF

Member
Joined
Jan 10, 2025
Messages
7
Programming Experience
10+
Folks:

I am gearing up for a project now in Proof of Concept stage. My specific issue is I have a component. It is simply a box. It is three deep from page so Page -> Major Component -> The Status Box in question. This is going to be a monitoring app. So - little to no user interaction. Changes come from the outside. (Like a fantasy football scoreboard.) UI change needs to occur based on events outside the application - identified by non-UI server side code.

My current task is simply to get the component to work. So I have put a button on the page component - and I kick the code from there. What I want to change is simply the css style background-color. The color is contained in a bind variable. After I change the color - I call StatusHasChanged(). The code seems to be working. I can breakpoint on the property get and observe is is executed after the button is pushed. Further - I can breakpoint on the StatusHasChanged and observe the color has indeed changed - as I expected. But the color never makes it to the UI. The background color of the box on the web client never changes.

Note - this is razor - not blazor. What I want to accomplish is very simple. I simply want the color of the box to change.

My .razor is:
HTML:
@inherits ForkStatusBoxBase


<style>
    .box {
        width: 25px;
        height: 25px;
        vertical-align: middle;
        background-color: @BackgroundColor;
        border: 2px solid black;
        padding: 5px;
        margin: 5px;
    }
</style>



<div class="box" @bind="BackgroundColor" />



My .razor.cs code behind is
C#:
public class ForkStatusBoxBase : ComponentBase
{
    string backgroundColor = "lightblue";




    public string BackgroundColor
    {
        get { return backgroundColor; }
        set { backgroundColor = value; }
    }

    public void GrantedFork()
    {

        backgroundColor = "green";
        StateHasChanged();

    }

}
Any help would be appreciated.
 
Last edited by a moderator:
I guess I'm learning something new. I didn't know that changing a backend value in Razor would cause the web client to re-download and re-render the web page. I always thought that was a Blazor thing rather than a Razor thing.

Have you watched the network traffic to see if the web browser gets an update from the server? Does the updated HTML have the new values?
 
I guess I'm learning something new. I didn't know that changing a backend value in Razor would cause the web client to re-download and re-render the web page. I always thought that was a Blazor thing rather than a Razor thing.

Have you watched the network traffic to see if the web browser gets an update from the server? Does the updated HTML have the new values?

Hi Skydiver:
I'm the one learning something new here. I have experience in WinFroms development. I'm trying to add to that skill set. I have come to understand that Razor and Blazor are practically synonymous. I'm focusing just on Razor following the principle - you learn c# before .Net. :)

I did look at network traffic. Nothing flows to the browser. To be clear I am calling Microsoft.AspNetCore.Components.ComponentBase.StateHasChanged. My expectation is only the component will be rerendered - not the entire page.
 
Hi Skydiver:
I'm the one learning something new here. I have experience in WinFroms development. I'm trying to add to that skill set. I have come to understand that Razor and Blazor are practically synonymous. I'm focusing just on Razor following the principle - you learn c# before .Net. :)

I did look at network traffic. Nothing flows to the browser. To be clear I am calling Microsoft.AspNetCore.Components.ComponentBase.StateHasChanged. My expectation is only the component will be rerendered - not the entire page.

One more observation. I put a Break Point on StateHasChanged in my .razor.cs file. On the break - I then F7. I find myself in the .razor file and the style has changed - the color has been updated as I expected. But the box (<div>) is not picking up the change in style and rerendering.
 
When I learned to use Razor, it was still in the early beta stages. It was MVC with Razor at that time. For any updates on the web browser to happen, there had to be some manually written JavaScript that would do an AJAX call back to the server and have it serve up some partial content -- basically something like what the WebForms Update panel does. I don't know what the current state of the art is for Razor.

As for your debugging steps, recall that the debugger does a dynamic request for the data when inspect values or use the immediate window. I think that you also need the web browser to request that same updated data.
 
Last edited:
The problem with this was the "Bind" statement in the razor file. Typical "newbe" trying to learn something, you pick something up, install it, test it, it does not work, then you leave it behind. I appreciate skydivers suggestion of looking at browser traffic. I have added that to my bag of tricks. Also, I see a very minimal network flow when I want to change the style.

The below works. What I like about it is it is the minimal - as simple as I can get it - solution. Many examples I came upon address very grandiose goals. I have always been a fan of Proof of Concept coding before a project. Get a skeleton that works, then flesh it out with business rules, error checking etc. That's worked for me!

Anyway - a minimal example of changing an attribute of style (it changes the background of a box) is:

.razor
HTML:
@inherits ForkStatusBoxBase


<style>
    .box {
        width: 25px;
        height: 25px;
        vertical-align: middle;
        background-color: @BackgroundColor;
        border: 2px solid black;
        padding: 5px;
        margin: 5px;
    }
</style>


<div class="box" style="background-color: @BackgroundColor" />




.razor.cs - the code behind
C#:
using System;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Components;


namespace xxxxxxxxxxxxxxxxx;


public class ForkStatusBoxBase  : ComponentBase
{
    string backgroundColor = "lightblue";


    public string BackgroundColor
    {
        get { return backgroundColor; }
        set { backgroundColor = value; }
    }

    public void GrantedFork()
    {

        backgroundColor = "green";
        StateHasChanged();

    }
}
 
Last edited by a moderator:
Ah, I see. So somehow a re-render happens when the style attribute is updated with the new value.
 
Ah, I see. So somehow a re-render happens when the style attribute is updated with the new value.

Not somehow. The component is updating when I call StateHasChanged - as it should. StateHasChanged sees the style change - and sends only that down to the browser.

The somehow ... going through my "learning experience" I introduced the "bind" idea to the code. Removing that - the thing works. So adding bind to the component somehow causes it NOT to update when StateHasChanged is called. I have not figured out why yet - I want to. But the issue is no longer impeding my progress so ....
 
FYI: when I posted the code above I had just gotten the component working. I was responding to a junk button I put on the UI. Thus - my code was being called from the UI Thread. My actual goal is to have the tool respond to events outside the UI. (Think Fantasy football scoreboard.) When I plumbed that in - I am now initiating the UI change from a thread other then the UI thread. The exception I got was very clear. I needed to change StateHasChanged() to InvokeAsync(StateHasChanged);. With that in place - the UI behaved as I wanted it to.
 
I have experience in WinFroms development

Maybe use Blazor then; as a long time winforms dev who'd never touched WPF, I found the jump logical and comfortable.

Also, I now use MudBlazor library for all my UIs and find the similarity to WinForms (in a modern MVC/data binding approach) greatly helps flatten the learning curve. I rarely touch JavaScript and only slightly more often use CSS
 
cjard - you are correct. A poor choice of words on my part. I am trying to learn Razor. I have learned that Blazor is based on Razor. So - Razor first. The effort I have been engaged on (and discussed here in this thread) is an academic (and to a small degree - Proof of Concept) code.

I have found that when you google ".net razor yada yada" you get a LOT of Blazor stuff. That is what I was trying to convey.

If I'm involved in the real world development effort - for sure we will use Blazor. I feel more comfortable understanding the base technology. I taught a bit of college in my day. I always tell students you cannot build a sexy house if you do not understand foundations.
 
Back
Top Bottom