Compare commits

..

1 Commits

5 changed files with 252 additions and 87 deletions
+173
View File
@@ -0,0 +1,173 @@
@import url('./../variables.css');
#preloader_container {
position: fixed;
width: 100%;
height: 100%;
top: 0;
left: 0;
background: rgba(0, 0, 0, 0.3);
z-index: 900;
backdrop-filter: blur(5px);
&.hide {
display: none;
}
&>#preloader{
width: 200px;
height: 200px;
background: none;
position: relative;
top: 30%;
left: 0;
right: 0;
margin-left: auto;
margin-right: auto;
&>.circle {
position: absolute;
border-radius: 50%;
box-sizing: border-box;
border: 10px solid transparent;
}
&>#circle_one {
width: 80%;
height: 80%;
border-top-width: 10px;
left: 5%;
top: 5%;
animation: roll_one 4.5s;
animation-fill-mode: forwards;
animation-iteration-count: infinite;
animation-timing-function: linear;
&.preloader-emerald-circle{
border-top-color: var(--color_emerald_light);
}
&.preloader-ruby-circle{
border-top-color: var(--color_ruby_main);
}
&.preloader-graphite-circle{
border-top-color: var(--color_graphite_main);
}
}
&>#circle_two {
width: 70%;
height: 70%;
left: 10%;
top: 10%;
border-right-width: 10px;
animation: roll_two 2.5s;
animation-fill-mode: forwards;
animation-iteration-count: infinite;
animation-timing-function: linear;
&.preloader-emerald-circle{
border-right-color: var(--color_emerald_light);
}
&.preloader-ruby-circle{
border-right-color: var(--color_ruby_main);
}
&.preloader-graphite-circle{
border-right-color: var(--color_graphite_main);
}
}
&>#circle_three {
width: 60%;
height: 60%;
left: 15%;
top: 15%;
border-bottom-width: 10px;
animation: roll_three 3s;
animation-fill-mode: forwards;
animation-iteration-count: infinite;
animation-timing-function: linear;
&.preloader-emerald-circle{
border-bottom-color: var(--color_emerald_light);
}
&.preloader-ruby-circle{
border-bottom-color: var(--color_ruby_main);
}
&.preloader-graphite-circle{
border-bottom-color: var(--color_graphite_main);
}
}
&>#preloader_text {
width: 150%;
left: -25%;
font-size: 1.5rem;
position: absolute;
bottom: -50px;
text-align: center;
color: white;
}
&>#logo {
position: absolute;
background: none;
width: 40%;
height: 40%;
top: 25%;
left: 25%;
&>.logo_square {
position: absolute;
box-sizing: border-box;
}
&>#left-top {
width: 60%;
height: 70%;
top: 15%;
left: 20%;
border-left: 7px solid white;
border-top: 7px solid white;
border-top-left-radius: 45%;
}
&>#right-bottom {
width: 40%;
height: 50%;
top: 15%;
left: 40%;
border-right: 7px solid white;
border-bottom: 7px solid white;
border-bottom-right-radius: 45%;
border-bottom-left-radius: 10%;
}
}
}
}
@keyframes roll_one {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}
@keyframes roll_two {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(-360deg);
}
}
@keyframes roll_three {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(720deg);
}
}
-81
View File
@@ -1,81 +0,0 @@
import axios from 'axios';
/**
* @author dgavrilov
* Пока в этом скрипте реализован axios иaнтерцептор для api запросов, отправляемых с фронта. Своего рода посредник для совершения "пред" и "пост" действий/проверок во время отправки api запроса
*/
//ГАВРИЛОВ
//переименуй файл на axios
//Инстанс api контура
const api = axios.create({
baseURL: '/public/api/',
withCredentials: true,
headers: {
'content-type': 'application/json',
'Accept': 'application/json',
}
})
//Инстанс web контура
const web = axios.create({
baseURL: '/public/',
withCredentials: true,
headers: {
'content-type': 'application/json',
'Accept': 'application/json',
}
})
//Гаврилов.
//Перехват запросов. Добавить подстановку bearer token?
// api.interceptors.request.use();
let isRefreshing = false, //Флаг выполнения запроса на обновления токена. Пока он иеет значение true, остальные запросы, попадающие в очередь из за полученной 401 ошибки кладутся в массив queueRqsts
queueRqsts = []; //запросы, которые получили 401 ошибку, одновремменно кладутся в эту переменную в виде промимсов, чтобы быть вызванными позднее, когда будет результат обновления токена превого запроса
//Если api ответ возвращает 401 ошибку, отправляем web запрос на роут фонового обновления санктум токена, где происходит проверка сессии, срока жизни санктум токена и его обновление, при удовлетворяющих условиях
//Перехват ответов api
api.interceptors.response.use(
response => response,
async error => {
const originalRequest = error.config; //Параметры вызываемого запроса
// originalRequest._retry = true;
if (error.response?.status === 401) {
if (isRefreshing) {
return new Promise((resolve, reject) => {
queueRqsts.push({
resolve: () => resolve(api(originalRequest)),
reject
})
})
} else {
isRefreshing = true;
try {
let resultRefresh;
//ГАВРИЛОВ. запись данных формы в session storage при неудачном silent regresh token. Чтобы после перезагрузки страницы подтянуть данные из session storage в форму? тогда придется заморачиваться с компонентом подтягивания данных из session storage при рендеринге страницы
resultRefresh = await web.get('silent_token_refresh')
if (!resultRefresh.data.original.data.token_refresh) {
throw new Error('Сессия истекла');
}
queueRqsts.forEach(pending => {
pending.resolve();
})
return api(originalRequest);
} catch (error) {
queueRqsts.forEach(pending => pending.reject(error))
isRefreshing = false;
queueRqsts = [];
window.location.href = '/public/login';
} finally {
isRefreshing = false;
queueRqsts = [];
}
}
}
return true;
}
)
export default api;
@@ -0,0 +1,39 @@
import "./../../../css/components/preloader.css";
import React from "react";
export default function Preloader ( props: {visible: boolean, text?: string | null}) {
const {text = 'загрузка'} = props;
let preloaderColorArray: string[] = ['preloader-ruby-circle', 'preloader-emerald-circle', 'preloader-graphite-circle'],
//Код ниже реализует возможность присваивать каждую перезагрузку разные классы с цветом полос
//Рандомное число, по которому получим первый класс
firstColorIndex: number = Math.floor(Math.random() * 3),
firstCircleClass: typeof preloaderColorArray[number] = preloaderColorArray[firstColorIndex],
secondCircleClass: typeof preloaderColorArray[number],
thirdCircleClass: typeof preloaderColorArray[number];
//Удалим уже используемый класс цвета из массива
preloaderColorArray.splice(firstColorIndex, 1);
//Оставшиеся 2 класса распределеяем в зависимости от проостой проверки четности/нечетности
if (Math.floor(Math.random() * 2) % 2 === 0) {
secondCircleClass = preloaderColorArray[0];
thirdCircleClass = preloaderColorArray[1];
} else {
secondCircleClass = preloaderColorArray[1];
thirdCircleClass = preloaderColorArray[0];
}
return (
props.visible ?
<div id='preloader_container'>
<div id='preloader'>
<div className={'circle ' + firstCircleClass} id='circle_one'></div>
<div className={'circle ' + secondCircleClass} id='circle_two'></div>
<div className={'circle ' + thirdCircleClass} id='circle_three'></div>
<div id='logo'>
<div className='logo_square' id='right-bottom'></div>
<div className='logo_square' id='left-top'></div>
</div>
<div id='preloader_text'>{ text }</div>
</div>
</div>
: null
)
}
@@ -0,0 +1,40 @@
import React, { createContext, useState } from "react";
import Preloader from "../components/preloader/Preloader";
interface PreloaderProps{
setPreloaderVisible: (preloaderVisible: boolean) => void;
setPreloaderText: (preloaderText: string) => void;
}
export const PreloaderContext = createContext<PreloaderProps>({
setPreloaderVisible: () => {},
setPreloaderText: () => {},
});
export function PreloaderProvider({ children }){
const [visible, setVisible] = useState<boolean>(true);
const [text, setText] = useState<string>('Страница загружается');
function setPreloaderVisible(preloaderVisible: boolean){
setVisible(preloaderVisible);
}
function setPreloaderText(preloaderText: string){
setText(preloaderText);
}
let value = {
setPreloaderVisible: setPreloaderVisible,
setPreloaderText: setPreloaderText
}
return (
<PreloaderContext.Provider value={value}>
<Preloader
visible={visible}
text={text}
/>
{children}
</PreloaderContext.Provider>
)
}
-6
View File
@@ -1,13 +1,7 @@
<?php
use Illuminate\Support\Facades\Route;
use \App\Http\Controllers;
use App\Http\Middleware\AuthenticateMagic;
Route::get('/', function () {
return view('welcome');
});
//ГАВРИЛОВ. добавить without middleware AuthMagicApi?
//Фоновое обновление санктум токена, если api вернул 401 (санктум протух), а сессия еще "жива"
Route::get('/silent_token_refresh', [Controllers\LoginController::class, 'silentRefreshUserSanctumToken'])->withoutMiddleware([AuthenticateMagic::class]);