I have classic web application rendered on server. I want to create admin panel as single page application in React. I want to server admin panel from https://smyapp.example.com/admin/. I try to use create-react-app
but it assumes that i serve SPA from root URL. How should I configure create-react-app
to serve app from "admin"
subdirectory? In documentation I found "homepage"
property but if I properly understand it requires complete url. I can't give complete url because my app is deployed in few environments.
Configure the deploy settings. Select a default branch to deploy (you can choose the master branch or any other branch) and ensure that the build command is npm run build and the publish directory is /build . Click Deploy site, and your React app will be deployed on Netlify's remote server.
Use a dot for the path to create a React app in the current directory, e.g. npx create-react-app . or npx create-react-app . --template typescript for TypeScript projects. Make sure your folder name doesn't contain special characters, spaces or capital letters.
In addition to your requirements, I am adding mine:
I also had to implement it in Angular2+ project not long ago, I found it harder to implement in React then in Angular2+ where you are good to go with ng build --base-href /<project_name>/
. source
PUBLIC_URL
env variable to the value of your subdirectory, let use /subdir
for example. You can also put this variable into your .env.production
(in case you do not have that file you can check the doc)public/index.html
add the base element bellow, this is for static files like images.<base href="%PUBLIC_URL%/">
public/index.html
, if you have custom link
element, make sure theyre are prefixed with %PUBLIC_URL%
(like manifest.json
and favicon.ico
href).BrowserRouter
, you can add basename
prop: <BrowserRouter basename={process.env.PUBLIC_URL} />
Router
instead, because you need access to history.push
method, to programmatically change page, do the following:// history.tsx
import {createBrowserHistory} from 'history';
export default createBrowserHistory({ basename: process.env.PUBLIC_URL });
<!-- Where your router is, for me, App.tsx -->
<Router history={history}>
...
</Router>
<!-- "./assets/quotes.png" is also ok, but "/assets/quotes.png" is not -->
<img src="assets/quotes.png" alt="" />
background-image
links from scss to jsx/tsx files (note that you may not need to do that if you use css files):/*remove that*/
background-image: url('/assets/background-form.jpg');
<section style={{backgroundImage: `url('assets/background-form.jpg')`}}>
...
You should be done.
I preferred to use PUBLIC_URL
instead of homepage
in package.json
because I want to use env variable set on gitlab to set the subdir. Relevant resources about the subject:
PUBLIC_URL
override homepage
, and PUBLIC_URL
also take the domain name, if you provide one. If you set only homepage
, PUBLIC_URL
will be set to the value of homepage
.
If you do not want to use a base element in your index.html (I would not know why), you will need to append process.env.PUBLIC_URL
to every link yourself. Note that if you have react-router with a base element, but have not set basename
prop, you will get a warning.
Sass won't compile with an incorrect relative path. It also won't compile with correct relative path to your ../public/assets
folder, because of ModuleScopePlugin
restrictions, you can avoid the restriction by moving your image inside the src folder, I haven't tried that.
There seem to be no way of testing relative path in development mode (npm start). see comment
Finnaly, theses stackoverflow link have related issues:
For create-react-app v2 and react-router v4, I used the following combo to serve a production (staging, uat, etc) app under "/app":
package.json:
"homepage": "/app"
Then in the app entry point:
<BrowserRouter basename={process.env.PUBLIC_URL}>
{/* other components */}
</BrowserRouter>
And everything "just works" across both local-dev and deployed environments. HTH!
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With