Compare commits
7 Commits
347f467735
...
work
| Author | SHA1 | Date | |
|---|---|---|---|
| 906d6df5ae | |||
| ca0afdad2c | |||
| 0cf2044506 | |||
| 2a54b37ead | |||
| 9d9d68e8d4 | |||
| 207e22d4f6 | |||
| 6cad7d6ed6 |
15
Dockerfile
Normal file
15
Dockerfile
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
FROM python:3.11-slim-bullseye
|
||||||
|
|
||||||
|
WORKDIR /usr/src/app
|
||||||
|
|
||||||
|
ENV PYTHONDONTWRITEBYTECODE 1
|
||||||
|
ENV PYTHONUNBUFFERED 1
|
||||||
|
|
||||||
|
RUN pip install --upgrade pip
|
||||||
|
COPY ./requirements.txt /usr/src/app/requirements.txt
|
||||||
|
RUN pip install -r requirements.txt
|
||||||
|
|
||||||
|
# copy project
|
||||||
|
COPY . /usr/src/app/
|
||||||
|
|
||||||
|
CMD ["python","app.py"]
|
||||||
118
app.py
118
app.py
@@ -3,13 +3,15 @@ from flask import Flask, render_template, request, flash, redirect, url_for, sen
|
|||||||
from werkzeug.utils import secure_filename
|
from werkzeug.utils import secure_filename
|
||||||
import os
|
import os
|
||||||
import secrets
|
import secrets
|
||||||
from models import db, PrintedDetal
|
|
||||||
import logging
|
import logging
|
||||||
|
from models import db, PrintedDetal
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
logging.basicConfig(filename='app.log', encoding='utf-8', level=logging.DEBUG)
|
logging.basicConfig(filename='app.log', encoding='utf-8', level=logging.DEBUG)
|
||||||
|
|
||||||
|
|
||||||
app = Flask(__name__)
|
app = Flask(__name__)
|
||||||
|
|
||||||
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///app.db'
|
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///app.db'
|
||||||
app.config['UPLOAD_IMG_FOLDER'] = os.path.join('static', 'uploads', 'img')
|
app.config['UPLOAD_IMG_FOLDER'] = os.path.join('static', 'uploads', 'img')
|
||||||
app.config['UPLOAD_STL_FOLDER'] = os.path.join('static', 'uploads', 'stl')
|
app.config['UPLOAD_STL_FOLDER'] = os.path.join('static', 'uploads', 'stl')
|
||||||
@@ -18,8 +20,7 @@ app.config['UPLOAD_SRC_FOLDER'] = os.path.join('static', 'uploads', 'src')
|
|||||||
db.init_app(app)
|
db.init_app(app)
|
||||||
|
|
||||||
|
|
||||||
@app.route("/")
|
def getlldb():
|
||||||
def index():
|
|
||||||
|
|
||||||
allDetals = {}
|
allDetals = {}
|
||||||
|
|
||||||
@@ -27,22 +28,38 @@ def index():
|
|||||||
|
|
||||||
for detal in detalsQuery:
|
for detal in detalsQuery:
|
||||||
detals = {}
|
detals = {}
|
||||||
|
detals['id'] = detal.id
|
||||||
detals['img'] = detal.img
|
detals['img'] = detal.img
|
||||||
detals['stl'] = detal.stl
|
detals['stl'] = detal.stl
|
||||||
detals['src'] = detal.src
|
detals['src'] = detal.src
|
||||||
detals['isprinted'] = detal.isprinted
|
detals['isprinted'] = detal.isprinted
|
||||||
|
detals['date'] = detal.dateadd
|
||||||
|
|
||||||
allDetals[detal.id] = detals
|
allDetals[detal.id] = detals
|
||||||
|
|
||||||
logging.debug(allDetals)
|
return allDetals
|
||||||
|
|
||||||
return render_template("index.html", detals=allDetals)
|
|
||||||
|
@app.route("/")
|
||||||
|
def index():
|
||||||
|
|
||||||
|
getall = getlldb()
|
||||||
|
|
||||||
|
return render_template("index.html", detals=getall)
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/login')
|
||||||
|
def login():
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/logout')
|
||||||
|
def logout():
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
@app.route("/admin", methods=['GET', 'POST'])
|
@app.route("/admin", methods=['GET', 'POST'])
|
||||||
def admipage():
|
def admipage():
|
||||||
logging.debug("adminpage loaded")
|
|
||||||
|
|
||||||
return render_template("adminpage.html")
|
return render_template("adminpage.html")
|
||||||
|
|
||||||
|
|
||||||
@@ -62,35 +79,100 @@ def upload_file():
|
|||||||
if len(stl_file.filename) > 0:
|
if len(stl_file.filename) > 0:
|
||||||
filename = stl_file.filename.split('.')
|
filename = stl_file.filename.split('.')
|
||||||
stl_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
stl_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
||||||
stl_file.save(os.path.join(app.config['UPLOAD_STL_FOLDER'],stl_file.filename))
|
stl_file.save(os.path.join(
|
||||||
|
app.config['UPLOAD_STL_FOLDER'], stl_file.filename))
|
||||||
|
|
||||||
if len(src_file.filename) > 0:
|
if len(src_file.filename) > 0:
|
||||||
filename = src_file.filename.split('.')
|
filename = src_file.filename.split('.')
|
||||||
src_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
src_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
||||||
src_file.save(os.path.join(app.config['UPLOAD_SRC_FOLDER'], src_file.filename))
|
src_file.save(os.path.join(
|
||||||
|
app.config['UPLOAD_SRC_FOLDER'], src_file.filename))
|
||||||
|
|
||||||
if request.form.get('isprinted') == 'on':
|
if request.form.get('isprinted') == 'on':
|
||||||
isprinered = True
|
isprinered = True
|
||||||
else:
|
else:
|
||||||
isprinered = False
|
isprinered = False
|
||||||
|
|
||||||
logging.debug(image_file.filename)
|
|
||||||
logging.debug(stl_file.filename)
|
|
||||||
logging.debug(src_file.filename)
|
|
||||||
logging.debug(isprinered)
|
|
||||||
|
|
||||||
detal = PrintedDetal(img=image_file.filename,
|
detal = PrintedDetal(img=image_file.filename,
|
||||||
stl=stl_file.filename,
|
stl=stl_file.filename,
|
||||||
src=src_file.filename,
|
src=src_file.filename,
|
||||||
isprinted=isprinered)
|
isprinted=isprinered,
|
||||||
|
datead=datetime.now().strftime('%d.%m.%Y'))
|
||||||
db.session.add(detal)
|
db.session.add(detal)
|
||||||
db.session.commit()
|
db.session.commit()
|
||||||
|
|
||||||
return redirect(url_for('admipage'))
|
return redirect(url_for('admipage'))
|
||||||
|
|
||||||
@app.route('/download/<filename>', methods=['GET', 'POST'])
|
|
||||||
def download(filename):
|
@app.route('/editcard')
|
||||||
return send_file(app.config['UPLOAD_SRC_FOLDER']+'/'+ filename, as_attachment=True)
|
def editcadrd():
|
||||||
|
allDetals = {}
|
||||||
|
|
||||||
|
detalsQuery = PrintedDetal.query.all()
|
||||||
|
|
||||||
|
for detal in detalsQuery:
|
||||||
|
detals = {}
|
||||||
|
detals['id'] = detal.id
|
||||||
|
detals['img'] = detal.img
|
||||||
|
detals['stl'] = detal.stl
|
||||||
|
detals['src'] = detal.src
|
||||||
|
detals['isprinted'] = detal.isprinted
|
||||||
|
detals['date'] = detal.dateadd
|
||||||
|
|
||||||
|
allDetals[detal.id] = detals
|
||||||
|
return render_template('editcard.html', detals=allDetals)
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/editcard/<id>')
|
||||||
|
def editcard2(id):
|
||||||
|
data = PrintedDetal.query.get(id)
|
||||||
|
print(data.id)
|
||||||
|
return render_template('cardupdate.html', data=data.id)
|
||||||
|
|
||||||
|
|
||||||
|
@app.route('/updater', methods=['POST'])
|
||||||
|
def updater():
|
||||||
|
|
||||||
|
if request.method == 'POST':
|
||||||
|
id = request.form.get('dataid')
|
||||||
|
detal = PrintedDetal.query.get(id)
|
||||||
|
|
||||||
|
image_file = request.files['img']
|
||||||
|
stl_file = request.files['stl']
|
||||||
|
src_file = request.files['src']
|
||||||
|
|
||||||
|
if len(image_file.filename) > 0:
|
||||||
|
filename = image_file.filename.split('.')
|
||||||
|
image_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
||||||
|
image_file.save(os.path.join(
|
||||||
|
app.config['UPLOAD_IMG_FOLDER'], image_file.filename))
|
||||||
|
detal.img = image_file.filename
|
||||||
|
|
||||||
|
if len(stl_file.filename) > 0:
|
||||||
|
filename = stl_file.filename.split('.')
|
||||||
|
stl_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
||||||
|
stl_file.save(os.path.join(
|
||||||
|
app.config['UPLOAD_STL_FOLDER'], stl_file.filename))
|
||||||
|
detal.stl = stl_file.filename
|
||||||
|
|
||||||
|
if len(src_file.filename) > 0:
|
||||||
|
filename = src_file.filename.split('.')
|
||||||
|
src_file.filename = secrets.token_hex(15) + '.' + filename[-1]
|
||||||
|
src_file.save(os.path.join(
|
||||||
|
app.config['UPLOAD_SRC_FOLDER'], src_file.filename))
|
||||||
|
detal.src = src_file.filename
|
||||||
|
|
||||||
|
if request.form.get('isprinted') == 'on':
|
||||||
|
isprinered = True
|
||||||
|
else:
|
||||||
|
isprinered = False
|
||||||
|
|
||||||
|
detal.isprinted = isprinered
|
||||||
|
db.session.commit()
|
||||||
|
|
||||||
|
return redirect(url_for('editcadrd'))
|
||||||
|
|
||||||
|
return "123"
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
|||||||
@@ -10,3 +10,5 @@ class PrintedDetal(db.Model):
|
|||||||
stl = db.Column(db.String(3000))
|
stl = db.Column(db.String(3000))
|
||||||
src = db.Column(db.String(3000))
|
src = db.Column(db.String(3000))
|
||||||
isprinted = db.Column(db.Boolean())
|
isprinted = db.Column(db.Boolean())
|
||||||
|
dateadd = db.Column(db.String(11))
|
||||||
|
|
||||||
BIN
requirements.txt
Normal file
BIN
requirements.txt
Normal file
Binary file not shown.
@@ -33,6 +33,10 @@ header {
|
|||||||
height: 450px;
|
height: 450px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.card-edit {
|
||||||
|
height: 400px;
|
||||||
|
}
|
||||||
|
|
||||||
.card-img-top{
|
.card-img-top{
|
||||||
|
|
||||||
height: 250px;
|
height: 250px;
|
||||||
|
|||||||
BIN
static/img/Auth.jpg
Normal file
BIN
static/img/Auth.jpg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 14 MiB |
56
templates/adduser.html
Normal file
56
templates/adduser.html
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css" rel="stylesheet"
|
||||||
|
integrity="sha384-Zenh87qX5JnK2Jl0vWa8Ck2rdkQ2Bzep5IDxbcnCeuOxjzrPF/et3URy9Bv1WTRi" crossorigin="anonymous">
|
||||||
|
<link rel="stylesheet" href="{{ url_for('static',filename='css/style.css')}}">
|
||||||
|
<title>3d Detals</title>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<header>
|
||||||
|
<h1 class="h1header"> Каталог печатных 3д деталей </h1>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="container">
|
||||||
|
<div class="row">
|
||||||
|
<div class="col-md-4 d-flex aligns-items-center mx-auto">
|
||||||
|
<div class="card card-login mobile-margin">
|
||||||
|
<img class="card-img-top" src="{{url_for('static', filename='img/Auth.jpg')}}">
|
||||||
|
|
||||||
|
|
||||||
|
<form method="POST" action="/auth">
|
||||||
|
<div class="col mb-3">
|
||||||
|
<label for="exampleInputEmail1" class="form-label">Введите логин</label>
|
||||||
|
<input type="text" class="form-control" name="login">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="col mb-3">
|
||||||
|
<label>Введите пароль</label>
|
||||||
|
<input type="password" class="form-control" name="passwd">
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<button type="submit" class="btn btn-primary">Авторизоваться</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/js/bootstrap.bundle.min.js"
|
||||||
|
integrity="sha384-OERcA2EqjJCMA+/3y+gxIOqMEjwtxJY7qPCqsdltbNJuaOe923+mo//f6V8Qbsw3"
|
||||||
|
crossorigin="anonymous"></script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
40
templates/cardupdate.html
Normal file
40
templates/cardupdate.html
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css" rel="stylesheet"
|
||||||
|
integrity="sha384-Zenh87qX5JnK2Jl0vWa8Ck2rdkQ2Bzep5IDxbcnCeuOxjzrPF/et3URy9Bv1WTRi" crossorigin="anonymous">
|
||||||
|
<link rel="stylesheet" href="{{ url_for('static', filename='css/adminpage.css')}}">
|
||||||
|
<title>3d Detals adminka</title>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<header>
|
||||||
|
<h1 class="h1header"> Обновление карточки {{data['id']}}</h1>
|
||||||
|
</header>
|
||||||
|
<div class="container">
|
||||||
|
|
||||||
|
<div class="row ">
|
||||||
|
<form method="post" enctype=multipart/form-data action="/updater">
|
||||||
|
<label class=" d-block p-2">Выберите изображение </label>
|
||||||
|
<input class=" formblock d-block p-2" type="file" name="img" />
|
||||||
|
<label class=" d-block p-2">Выберите исходный file </label>
|
||||||
|
<input class=" formblock d-block p-2" type="file" name="src" />
|
||||||
|
<label class=" d-block p-2">Выберите stl file </label>
|
||||||
|
<input class=" formblock d-block p-2" type="file" name="stl" />
|
||||||
|
<label > Напечатано </label>
|
||||||
|
<input type="checkbox" class="bxigchekcbox" id="printed" name="isprinted">
|
||||||
|
<input class="btn d-block p-2" type="submit" value="Обновить">
|
||||||
|
<input name="dataid" type="hidden" value="{{data}}" >
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/js/bootstrap.bundle.min.js"
|
||||||
|
integrity="sha384-OERcA2EqjJCMA+/3y+gxIOqMEjwtxJY7qPCqsdltbNJuaOe923+mo//f6V8Qbsw3"
|
||||||
|
crossorigin="anonymous"></script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
48
templates/editcard.html
Normal file
48
templates/editcard.html
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css" rel="stylesheet"
|
||||||
|
integrity="sha384-Zenh87qX5JnK2Jl0vWa8Ck2rdkQ2Bzep5IDxbcnCeuOxjzrPF/et3URy9Bv1WTRi" crossorigin="anonymous">
|
||||||
|
<link rel="stylesheet" href="{{ url_for('static',filename='css/style.css')}}">
|
||||||
|
<title>3d Detals</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<header>
|
||||||
|
<h1 class="h1header"> Каталог печатных 3д деталей </h1>
|
||||||
|
</header>
|
||||||
|
<div class="container">
|
||||||
|
<div class="row">
|
||||||
|
|
||||||
|
{% for key, value in detals.items() %}
|
||||||
|
|
||||||
|
<div class="col-md-4">
|
||||||
|
<div class="card card-edit mobile-margin">
|
||||||
|
<img class="card-img-top" src="{{url_for('static', filename='uploads/img/' + value['img'])}}">
|
||||||
|
|
||||||
|
<div class="card-body">
|
||||||
|
<dv class="row">
|
||||||
|
<div class="col">
|
||||||
|
<a href="/editcard/{{value['id']}}" class="btn btn-block mobile-margin"> Изменить </a>
|
||||||
|
</div>
|
||||||
|
<div class="row">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{%endfor%}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/js/bootstrap.bundle.min.js"
|
||||||
|
integrity="sha384-OERcA2EqjJCMA+/3y+gxIOqMEjwtxJY7qPCqsdltbNJuaOe923+mo//f6V8Qbsw3"
|
||||||
|
crossorigin="anonymous"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -25,6 +25,9 @@
|
|||||||
|
|
||||||
<div class="card-body">
|
<div class="card-body">
|
||||||
<dv class="row">
|
<dv class="row">
|
||||||
|
<div class="col">
|
||||||
|
<a href="{{url_for('static', filename='uploads/img/' + value['img'])}}" class="btn btn-block mobile-margin" download>Скачать изображение </a>
|
||||||
|
</div>
|
||||||
<div class="col">
|
<div class="col">
|
||||||
<a href="{{url_for('static', filename='uploads/src/' + value['src'])}}" class="btn btn-block mobile-margin" download >Скачать исходник</a>
|
<a href="{{url_for('static', filename='uploads/src/' + value['src'])}}" class="btn btn-block mobile-margin" download >Скачать исходник</a>
|
||||||
</div>
|
</div>
|
||||||
@@ -33,11 +36,11 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col d-flex justify-content-center">
|
<div class="col d-flex justify-content-center">
|
||||||
<input class="form-check-input" type="checkbox" value="" id="flexCheckDefault" disabled>
|
<input class="form-check-input" type="checkbox" {{'checked="checked"' if value['isprinted']==1 else ""}} id="flexCheckDefault" disabled>
|
||||||
<label> Напечатано </label>
|
<label> Напечатано </label>
|
||||||
</div>
|
</div>
|
||||||
<div class="col d-flex justify-content-center">
|
<div class="col d-flex justify-content-center">
|
||||||
<h5> 22.10.2022</h5>
|
<h5> {{value['dateadd']}}</h5>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
55
templates/login.html
Normal file
55
templates/login.html
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/css/bootstrap.min.css" rel="stylesheet"
|
||||||
|
integrity="sha384-Zenh87qX5JnK2Jl0vWa8Ck2rdkQ2Bzep5IDxbcnCeuOxjzrPF/et3URy9Bv1WTRi" crossorigin="anonymous">
|
||||||
|
<link rel="stylesheet" href="{{ url_for('static',filename='css/style.css')}}">
|
||||||
|
<title>3d Detals</title>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<header>
|
||||||
|
<h1 class="h1header"> Каталог печатных 3д деталей </h1>
|
||||||
|
</header>
|
||||||
|
|
||||||
|
<div class="container container-login d-flex h-100 ">
|
||||||
|
<div class="row d-flex justify-content-center ">
|
||||||
|
<div class="col-md-4">
|
||||||
|
<div class="card align-middle mobile-margin">
|
||||||
|
|
||||||
|
<div class="card-body text-center">
|
||||||
|
<dv class="row">
|
||||||
|
<h2 class="fw-bold mb-2 text-uppercase">Авторизация</h2>
|
||||||
|
<p class="mb-5">Введите логин и пароль</p>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="form-outline form-white mb-4">
|
||||||
|
<input type="email" id="typeEmailX" class="form-control form-control-lg" />
|
||||||
|
<label class="form-label" for="typeEmailX">Email</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-outline form-white mb-4">
|
||||||
|
<input type="password" id="typePasswordX" class="form-control form-control-lg" />
|
||||||
|
<label class="form-label" for="typePasswordX">Password</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="col">
|
||||||
|
<a href="" class="btn btn-block mobile-margin" download>Авторизоваться </a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.2.2/dist/js/bootstrap.bundle.min.js"
|
||||||
|
integrity="sha384-OERcA2EqjJCMA+/3y+gxIOqMEjwtxJY7qPCqsdltbNJuaOe923+mo//f6V8Qbsw3"
|
||||||
|
crossorigin="anonymous"></script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
Reference in New Issue
Block a user