Skip to content

Latest commit

 

History

History
191 lines (152 loc) · 8.64 KB

README-V2.md

File metadata and controls

191 lines (152 loc) · 8.64 KB

NOTE: This README is outdated, see root of repo for latest

BREAKING CHANGES: Breaking changes how to commit

BlazorTransitionableRoute V2.1.0

Allows current and previous route to exist enabling transition animations of UI/UX design systems.

What it does?

Sometimes you need to show transitions between route but Blazor by default only allows one. This provides an ability to remember the last route and allow you to perform transitions out and in on the old and new route views rendered.

Demo

How it works?

Having two view layouts means we can remember the previous route but we cannot simply overwrite them, they must be preserved so we do not lose browser state like scroll position etc. This solution handles the route views and knows when to call the transition implementation you provide to it. It also works with the browser navigation buttons. If you need to handle in app back buttons then use the jsInterop to call the native back i.e. window.history.back(); This is as simple a solution I could arrive at. If there is another simpler one please let me know.

Version 2.1.0

Addition of ForgetStateOnTransition option to reset page state when returning to previous route via navigation.

Version 2.0.0

This version simplifies the implemention of this component and also your usage of it. You can now implement transition behaviour directly with Blazor code and not having to rely on JavaScript Interop to do so. You can also adjust transition behaviour on first render via Transition.FirstRender and you can get the direction from Transition.Backwards.

Breaking changes to V1

  • TransitioningIn is now Transition.IntoView (Transition now provides more detail including direction and if first render).
  • Remove DI for BlazorTransitionableRoute.NavigationState and BlazorTransitionableRoute.NavigationStateHandler as these are no longer present.
    • At minimum you need this DI registration: IRouteTransitionInvoker implemented by DefaultRouteTransitionInvoker.
  • IRouteTransitionInvoker replaces parameter of BrowserNavigationDirection for Backwards (bool) value.

for v1 documentation see here

Roadmap

  1. Add a property (could be previous route data) allowing custom transitions for more complex UI, for example master/detail transitions. Your implementation could then be a simple registry of transition type combinations mapped to animation styles.
  2. Update demos and library to latest Blazor version.
  3. Major change to a living route history implementation allowing deeper navigation.

Demos

See the demos for how to implement.

  • The BlazorTransitionableRouteDemoWasm demo shows a Blazor coded transition behaviour.
  • The BlazorTransitionableRouteDemoServer demo shows a JavaScript Interop transition behaviour.

Both methods are interchangeable for Wasm and Server.

Installation

Latest version in here: NuGet

Install-Package BlazorTransitionableRoute

or

dotnet add package BlazorTransitionableRoute

Usage

Common component steps

For client-side and server-side Blazor - add script section to index.html or _Host.cshtml (head section)

<script src="_content/BlazorTransitionableRoute/jsInterop.js"></script>

Add reference to _Imports.razor file

@using BlazorTransitionableRoute

For client-side and server-side Blazor - add registrations for the dependency injection framework in Program.cs or Startup.cs, for example

builder.Services.AddScoped<BlazorTransitionableRoute.IRouteTransitionInvoker, MyRouteTransitionInvoker>();

At a minimum if you implement transition behaviour via Blazor code and not custom JavaScript interop then you need to register the BlazorTransitionableRoute.DefaultRouteTransitionInvoker.

A change to routing approach

Modify the App.razor file to take advantage of the transitionable route layouts and view. This means moving the MainLayout to be more explicit in the app router and providing a more container like MyViewLayout as the default layouts. You can see below the simple use of primary and secondary route views. The TransitionableRoutePrimary / Secondary modify the RouteData passed to each inner TransitionableRouteView based on the active state, which is swapped after each navigation to preserve component instances.

<Router AppAssembly="@typeof(Program).Assembly">
    <Found Context="routeData">
        <LayoutView Layout="@typeof(MainLayout)">
            <TransitionableRoutePrimary RouteData="@routeData" ForgetStateOnTransition="false">
                <TransitionableRouteView DefaultLayout="@typeof(MyViewLayout)" />
            </TransitionableRoutePrimary>
            <TransitionableRouteSecondary RouteData="@routeData" ForgetStateOnTransition="false">
                <TransitionableRouteView DefaultLayout="@typeof(MyViewLayout)" />
            </TransitionableRouteSecondary>
        </LayoutView>
    </Found>
    <NotFound>
        <LayoutView Layout="@typeof(MainLayout)">
            <p>Sorry, there's nothing at this address.</p>
        </LayoutView>
    </NotFound>
</Router>

Example usage

You will need to create your own transitiong view, for example (Transition parameter is provided by the inherited TransitionableLayoutComponent)

@inherits TransitionableLayoutComponent

<div class="@transitioningClass">
    @Body
</div>

@code {
    private string transitioningDirection => Transition.Backwards ? "Up" : "Down";

    private string transitioningClass => Transition.FirstRender ? "" : Transition.IntoView
        ? $"animate__fadeIn{transitioningDirection} animate__faster animate__animated"
         : $"animate__fadeOut{transitioningDirection} animate__faster animate__animated";
}

(alternatively you can use the default one provided by the component called TransitionableLayoutComponent but you will need to handle the `Transition' cascading parameter and probably wrap each page in it's own containing component. You are free to implement how you like but the cascading parameter is your starting point to prepare for transitioning.)

Optional example JavaScript Interop usage

You can optionally create an implementation of IRouteTransitionInvoker and save it where you like, perhaps in Shared folder and make sure it is registered with DI.

using BlazorTransitionableRoute;
using Microsoft.JSInterop;
using System.Threading.Tasks;

namespace BlazorTransitionableRouteDemoWasm.Client.Shared
{
    public class RouteTransitionInvoker : IRouteTransitionInvoker
    {
        private readonly IJSRuntime jsRuntime;

        public RouteTransitionInvoker(IJSRuntime jsRuntime)
        {
            this.jsRuntime = jsRuntime;
        }

        public async Task InvokeRouteTransitionAsync(bool backwards)
        {
            await jsRuntime.InvokeVoidAsync("window.yourJsInterop.transitionFunction", backwards);
        }
    }
}

For client-side and server-side Blazor - add script section to index.html or _Host.cshtml (head section), for example

<script src="yourJsInteropForAnimatedTransitions.js"></script>
... any other supporting animation library scripts you are using, for example using animate.css
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/animate.css/4.0.0/animate.min.css" />

Add your js interop implementation, for example (this is from the demos using animate.css)

window.yourJsInterop = {
    transitionFunction: function (back) {

        let transitionIn = document.getElementsByClassName('transition-in')[0];
        let transitionOut = document.getElementsByClassName('transition-out')[0];

        let direction = back ? "Up" : "Down";

        if (transitionIn && transitionOut) {

            transitionOut.classList.remove('transition-out');
            transitionOut.classList.add(
                "animate__fadeOut" + direction,
                "animate__faster",
                "animate__animated"
            );

            transitionIn.classList.remove('transition-in');
            transitionIn.classList.add(
                "animate__fadeIn" + direction,
                "animate__faster",
                "animate__animated"
            );
        }
    }
}

If you experience timing issues when transition animations are not performing you may need to wrap you inner function code in a zero timeout, for example

window.yourJsInterop = {
    transitionFunction: function (back) {
        setTimout(() => {
           ...
        }, 0);
    }
}