我使用reactjs作为前端,django作为后端。 React路由器用于路由。当我刷新由反应路由器路由的页面时,我得到django 404 Page Not Found error
。如果我刷新主页,我不会得到任何这样的错误,因为主页也是由django模板使用其url呈现。
我是否必须在webpack中配置它?我的项目结构是我已经分离了django和reactjs。我创建了一个文件夹作为reactjs文件所在的前端。
更新
主页模板包含addrestaurant等路线的所有链接。
我的webpack.config文件
const path = require("path");
if(!process.env.NODE_ENV) {
process.env.NODE_ENV = 'development';
}
module.exports = {
entry: [
'./src/index.js'
],
output: {
path: path.join("../app/static/build/", "js"),
filename: "app.js",
publicPath: "../app/static/build/"
},
devtoo: 'source-map',
debug: true,
module: {
loaders: [{
exclude: /node_modules/,
loader: 'babel',
query: {
presets: ['react', 'es2015', 'stage-1']
}
},
{test: /\.(jpe?g|png|gif|svg)$/i, loader: "url-loader?name=images/[name].[ext]"},
]
},
resolve: {
extensions: ['', '.js', '.jsx']
},
devServer: {
historyApiFallback: true,
contentBase: './'
}
};
urls.py
urlpatterns = [
url(r'^', views.home, name="homePage"),
url(r'^(?:.*)/?$', views.home),
]
home.html的
{% extends 'base.html' %}
{% block title %} Foodie | Homepage {% endblock title%}
{% block content %}
<div class="homepage">
</div>
{% endblock %}
{% block js %}
{{ block.super }}
<script type="text/javascript">
var data = {
isUserAuthenticated:{% if request.user.is_authenticated %}true{% else %}false{% endif %}
};
console.log('data',data);
$(function() {
app.showHomePage(".homepage",data);
});
</script>
{% endblock %}
index.js
window.app = {
showHomePage: function(id,data){
render(
<Provider store={createStoreWithMiddleware(reducers)}>
<Router>
<App />
</Router>
</Provider>, document.querySelector(id)
);
},
}
Banner是App组件的子组件
const Banner = (props) => (
<div className="navbar-container">
<div className="ui container">
<div className="ui large secondary menu">
<a className="toc item">
<i className="sidebar icon"></i>
</a>
<div className="item logo">
<div className="ui logo shape">
<div className="sides">
<div className="active ui side">
Foodie
</div>
</div>
</div>
</div>
<Link to="/restaurant" className="active item tab">Home</Link>
<Link to='/addrestaurant' className='item tab'>Add Restaurant</Link>
<Link to="/products" className="item tab">Products</Link>
<div className="right item">
<a href="" id="bookingInfoButton" className="ui white inverted button">Booking</a>
</div>
</div>
</div>
</div>
);
export default Banner;
答案 0 :(得分:15)
问题可能是您尚未配置URL来处理React Router中定义的路由。在您的Django urls.py
中,您应该使用catch all来匹配索引模板的所有URL
urlpatterns += [
# match the root
url(r'^$', base_view),
# match all other pages
url(r'^(?:.*)/?$', base_view),
]
base_view
是一个视图功能,可以呈现包含您的捆绑应用的模板。
答案 1 :(得分:2)
万一有人想知道,我有确切的问题,Paul S的答案解决了。添加答案而不是评论仅是因为SO不允许我格式化评论中的代码片段。我最终在path()
中混合使用了django的新urls()
和旧的urls.py
:
urlpatterns = [
path('login/', LoginView.as_view(), name='login'),
path('logout/', LogoutView.as_view()),
path('/', login_required(TemplateView.as_view(template_name="app.html"),
login_url='login')),
url(r'^(?:.*)/?$', login_required(TemplateView.as_view(template_name="app.html"),
login_url='login')),
]
Django处理登录,注销和根/
。 React Router可以处理其他一切
答案 2 :(得分:2)
万一有人遇到同样的问题,在django 2.0中,请按照'Kevin Martin Jose'的答案进行操作,而应将 url 替换为 re_path
from django.urls import path, re_path
urlpatterns = [
path('login/', LoginView.as_view(), name='login'),
path('logout/', LogoutView.as_view()),
path('/', login_required(TemplateView.as_view(template_name="app.html"),
login_url='login')),
re_path(r'^(?:.*)/?$', login_required(TemplateView.as_view(template_name="app.html"),
login_url='login')),
]
答案 3 :(得分:1)
这是受 Kevin Jones 和 Paul S 回答启发的回答。我在使用正则表达式和托管 REST API 时遇到问题。如果我的前端应用程序在进行 API 调用时没有附加斜杠,它就不会匹配并且会被路由回前端。这是因为 django 设置 APPEND_SLASH=True
要求它通过 urlpatterns
并在附加斜杠并再次尝试之前失败一次。因此,这里有一个正则表达式,它只排除以“api”或“admin”开头的任何内容,否则将其发送到前端。
urlpatterns = [
path("admin/", admin.site.urls),
path("api/", include(router.urls)), # from rest_framework
re_path('(^(?!(api|admin)).*$)',
TemplateView.as_view(template_name="build/index.html")),
]
答案 4 :(得分:0)
这是一种不会导致其他视图覆盖的解决方案,并且不需要您丢弃 404 处理程序。
缺点是这需要保持最新的路线列表,但这就是该文件的用途。
在urls.py
中:
from django.urls import re_path
from myapp import views
# This wrapper turns your list into a regex URL matcher
react_views_regex = r'\/|\b'.join([
# List all your react routes here
'view1',
'view2'
]) + r'\/'
urlpatterns = [
# Change views.index to whatever your view is
re_path(react_views_regex, views.index),
]
这些 URL 应该使用或不使用斜杠,区分大小写(如真实路由),并且只会匹配完整的单词(因此像“user”和“users”这样的视图不会冲突)。