在 JS 中实现 Laravel 的 ROUTE 函数

在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几乎没有差别

本作品采用《CC 协议》,转载必须注明作者和本文链接
本帖由 Summer 于 7年前 加精
leo
《L05 电商实战》
从零开发一个电商项目,功能包括电商后台、商品 & SKU 管理、购物车、订单管理、支付宝支付、微信支付、订单退款流程、优惠券等
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
讨论数量: 20
leo

@oustn 你要的文章来了

7年前 评论
leo
7年前 评论

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

7年前 评论
Summer

之前有个哥们跟我说过

leo 很深

leo 确实很深

7年前 评论
Destiny

很深。。

7年前 评论

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>
7年前 评论
leo

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

7年前 评论

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

7年前 评论
leo

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

7年前 评论

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

7年前 评论
jcc123

注2:这里引用了lodash

原来 loadsh 在这里

leo 大神的文章得研究

在每一处的不经意间就是一个问题的解法:100:

6年前 评论
leo

@Asa_c 新版本的 Laravel 将 getUri() 重命名为 uri()

5年前 评论
monkey

可以考虑这个 扩展包:https://github.com/tightenco/ziggy

4年前 评论

@leo 大佬, 我这里出现了个小问题
/resouces/js/bootstrap.js文件中引入route方法, 在前端调用时显示未定义. 也运行npm run watch-poll了, 而且在调试模式下的app.js文件中可以搜到route方法, 但就是报route is not a function
file
file

4年前 评论
leo (楼主) 4年前
_mars (作者) 4年前
leo (楼主) 4年前
_mars (作者) 4年前

提示route_uri未定义。Call to undefined function route_uri() (View:

3年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!