在 JS 中实现 Laravel 的 ROUTE 函数

分享 leo ⋅ 于 2016-12-16 18:17:26 ⋅ 最后回复由 839891627 2016-12-19 14:31:45 ⋅ 783 阅读

在Laravel的路由模块里,我们可以给每一个路由设定一个名字,比如:

Route::get('/blog/{blog}', 'BlogController@show')->name('blog.show');

然后就可以通过 route('blog.show', ['blog' => 1]) 来获取到这个路由的访问地址,后端跳转可以用

return redirect()->route('blog.show', ['blog' => 1]);

这样做的好处是如果发生url变更,比如我想把/blog/{blog}改成/boke/{blog},只需要改路由文件,别的地方都不用调整。所以推荐给每个路由都命个名,并通过路由名的方式来获取。

但这个仅限于后端以及blade模板可以使用,稍微上点规模的网站都会把js文件单独拎出来,不会把js直接写在blade模板中,这样就导致js里发ajax请求时或者页面跳转时只能将请求地址写死,比如

location.href = '/blog/' + id;

这样万一路由发生变更,还得去修改js文件,如果同一个路由被多个js调用,很容易漏改那么一两个。

所以我就考虑能不能在js中实现一个类似后端的route函数。

最终的解决方案很简单,两个函数就可以搞定。

后端部分需要实现一个函数:

function route_uri($name)
{
    return app('router')->getRoutes()->getByName($name)->getUri();
}

这个函数的作用是根据路由名称返回原始的路由地址,比如:

echo route_uri('blog.show'); 
// 会输出/blog/{blog}

前端也只需要一个函数:

let route = (routeUrl, param) => {
    let append = [];

    for (let x in param) {
        let search = '{' + x + '}';

        if (routeUrl.indexOf(search) >= 0) {
            routeUrl = routeUrl.replace('{' + x + '}', param[x]);
        } else {
            append.push(x + '=' + param[x]);
        }
    }

    let url = '/' + _.trimStart(routeUrl, '/');

    if (append.length == 0) {
        return url;
    }

    if (url.indexOf('?') >= 0) {
        url += '&';
    } else {
        url += '?';
    }

    url += append.join('&');

    return url;
}
  • 注1:ES6语法,如果没用babel的自行转ES5
  • 注2:这里引用了lodash

这个函数的作用是:

route('/blog/{blog}', {blog: 1}); //返回 /blog/1
route('/blog/{blog}', {blog: 1, preview: 1}); //返回 /blog/1?preview=1

然后就很简单了,在blade模板中把这个页面所有需要用到的路定义在一个对象中:

var routes = {
    blog: {
        show: '{{ route_uri('blog.show') }}',
        update: '{{ route_uri('blog.update') }}',
        destroy: '{{ route_uri('blog.destroy') }}'
    }
};

在js文件里就可以:

$.post(route(routes.blog.update, {blog: 1}), {title: 'xxx', content: 'xxx'})

用法和后端的route几乎没有差别

本帖已被设为精华帖!
本帖由 Summer 于 1月前 加精
回复数量: 14
  • leo MOD 不会写前端的后端不是好运维
    2016-12-16 18:18:55

    @oustn 你要的文章来了

  • leo MOD 不会写前端的后端不是好运维
    2016-12-16 18:21:20
  • oustn 没什么好介绍的
    2016-12-16 18:36:19

    哦 很详细,还支持 uri search 参数。 :+1:

  • Summer MOD A Life-long learner.
    2016-12-16 21:27:03

    之前有个哥们跟我说过

    leo 很深

    leo 确实很深

  • KevinYang
    2016-12-16 22:51:53

    学习了

  • jw嘉华 j 了个 w
    2016-12-16 23:04:10

    mark

  • Destiny 将来的你一定会感谢现在努力的自己!
    2016-12-16 23:21:26

    很深。。

  • mingyun
    2016-12-18 21:35:27

    厉害了

  • killtw
    2016-12-19 10:01:16

    Laroute

    推薦使用

  • 839891627
    2016-12-19 14:21:33

    es6不了解。其实我想知道的这种情况,路由里面定义了参数,js里面获取的参数怎么拼接进去:

    Route::get('user/{id}/update');
    
    //route('user.update', $user->id);

    那么在js里面这个url要怎么生成?

    <script>
    var uid = $('#foo').val();
    var url = "{{ route('user.update') }}"。。。  // 这里的 uid 怎么拼接进去?
    
    </script>
  • leo MOD 不会写前端的后端不是好运维
    2016-12-19 14:24:17

    @839891627 用我的route_uri函数,uid的拼接放在前端的route函数里

  • 839891627
    2016-12-19 14:25:59

    @leo 如何不用呢:laughing: 怎么写这个路由

  • leo MOD 不会写前端的后端不是好运维
    2016-12-19 14:26:47

    @leo 如果不用……您就另寻他法吧,我也不知道

  • 839891627
    2016-12-19 14:31:45

    @leo 还是学习了。没找到解决之前,先用着:smile: 另外你的id,5320,让我想到nokia5320 :notes:

暂无评论~~
  • 请注意单词拼写,以及中英文排版,参考此页
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`, 更多语法请见这里 Markdown 语法
  • 支持表情,使用方法请见 Emoji 自动补全来咯,可用的 Emoji 请见 :metal: :point_right: Emoji 列表 :star: :sparkles:
  • 上传图片, 支持拖拽和剪切板黏贴上传, 格式限制 - jpg, png, gif
  • 发布框支持本地存储功能,会在内容变更时保存,「提交」按钮点击时清空
Ctrl+Enter