Loading html and Controller from server and creating dynamic states UI - router

前端 未结 4 1912
后悔当初
后悔当初 2021-02-05 11:34

I am looking for a Solution to load my App Content dynamically from the Server.

My Scenario:

Lets say we have 2 Users (A and B), my App consists of different Mod

4条回答
  •  醉话见心
    2021-02-05 12:00

    I think I'm doing what you're asking. I achieve this by using UI-Router, ocLazyLoad and ui-routers future states. Essentially our setup allows us to have 50+ modules, all in the same code base, but when a user opens the app. its starts by only loading the base files required by the app. Then, as the user moves around between states, the application will load up the files required for that part, as their needed. (apologies for the fragmented code, I've had to rip it out of the code base, but tried to only provide the stuff thats actually relevant to the solution).

    Firstly, the folder structure

    • Core App

    config.js

    • Module 1 (/module1)

    module.js

    controllers.js

    • Module 2 (/module2)

    module.js

    controllers.js

    etc

    Config.js:

    The first thing we do is create the base state, this is an abstract state, so the user can never actually just hit it.

    $stateProvider.state('index', {
        abstract: true,
        url: "/index",
        views: {
            '': {
                templateUrl: "views/content.html" // a base template to have sections replaced via ui-view elements
            }
        },
        ...
    });
    

    Then we configure the modules in ocLazyLoad. This allows us to just tell ocLazyLoad to load the module, and it loads all the required files (although in this instance, its only a single file, but it allows each module to have varying paths).

    $ocLazyLoadProvider.config({
        loadedModules: ['futureStates'],
        modules: [
            {
                name: 'module1',
                files: ['module1/module.js']
            },
            {
                name: 'module2',
                files: ['module2/module.js']
            }
        ]
    });
    

    Next we create a function to allow ui-router to load the modules when requested (through future states).

    function ocLazyLoadStateFactory($q, $ocLazyLoad, futureState) {
        var deferred = $q.defer();
        // this loads the module set in the future state
        $ocLazyLoad.load(futureState.module).then(function () {
            deferred.resolve();
        }, function (error) {
            deferred.reject(error);
        });
        return deferred.promise;
    }
    $futureStateProvider.stateFactory('ocLazyLoad', ['$q', '$ocLazyLoad', 'futureState', ocLazyLoadStateFactory]);
    

    Then we configure the actual future states. These are states that may be loaded in the future, but we don't want to configure them right now.

    $futureStateProvider.futureState({
        'stateName': 'index.module1', // the state name
        'urlPrefix': '/index/module1', // the url to the state
        'module': 'module1', // the name of the module, configured in ocLazyLoad above
        'type': 'ocLazyLoad' // the future state factory to use.
    });
    $futureStateProvider.futureState({
        'stateName': 'index.module2',
        'urlPrefix': '/index/module2',
        'module': 'module2',
        'type': 'ocLazyLoad'
    });
    

    If you want the list of future states to be provided asynchronously:

    $futureStateProvider.addResolve(['$http', function ($http) {
        return $http({method: 'GET', url: '/url'}).then(function (states) {
            $futureStateProvider.futureState({
                'stateName': 'index.module2',
                'urlPrefix': '/index/module2',
                'module': 'module2',
                'type': 'ocLazyLoad'
            });
        });
    }]);
    

    Then we configure the modules as follows:
    module1/module.js

    $stateProvider.state('index.module1', {
        url: "/module1",
        abstract: true,
        resolve: {
            loadFiles: ['$ocLazyLoad', function($ocLazyLoad){
                return  return $ocLazyLoad.load(['list of all your required files']);
            }]
        }
    })
    $stateProvider.state('index.module1.sub1', {
        url: "/sub1",
        views: {
           // override your ui-views in here. this one overrides the view named 'main-content' from the 'index' state
           'main-content@index': {
                templateUrl: "module1/views/sub1.html"
            }
        }
    })
    

提交回复
热议问题