fix user keys layout, adjust @media breakpoints to bs4, fix more unused css, optimize script
This commit is contained in:
parent
18df2fd647
commit
4ebd52cd69
6 changed files with 162 additions and 108 deletions
|
@ -169,13 +169,13 @@
|
||||||
max-width: 1120px;
|
max-width: 1120px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
@media (max-width: 767px) {
|
||||||
.content-dashboard {
|
.content-dashboard {
|
||||||
padding: 0 15px;
|
padding: 0 15px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 540px) {
|
@media (max-width: 576px) {
|
||||||
select {
|
select {
|
||||||
width: 280px;
|
width: 280px;
|
||||||
}
|
}
|
||||||
|
@ -284,7 +284,7 @@
|
||||||
margin: 15px auto;
|
margin: 15px auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (min-width: 768px) and (max-width: 900px) {
|
@media (min-width: 768px) and (max-width: 991px) {
|
||||||
.modal-dialog {
|
.modal-dialog {
|
||||||
width: 50%;
|
width: 50%;
|
||||||
}
|
}
|
||||||
|
@ -296,7 +296,7 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media(min-width: 320px) {
|
@media(min-width: 576px) {
|
||||||
.modal:before {
|
.modal:before {
|
||||||
content: '';
|
content: '';
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
|
|
|
@ -215,7 +215,7 @@ textarea {
|
||||||
|
|
||||||
/* Show the dropdown menu on hover */
|
/* Show the dropdown menu on hover */
|
||||||
|
|
||||||
@media (min-width: 769px) {
|
@media (min-width: 768px) {
|
||||||
.nav-language .dropdown:hover .dropdown-menu {
|
.nav-language .dropdown:hover .dropdown-menu {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
@ -769,7 +769,7 @@ tech-sub-sec h2 {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
@media (max-width: 767px) {
|
||||||
.ssdimg img {
|
.ssdimg img {
|
||||||
width: 100px;
|
width: 100px;
|
||||||
}
|
}
|
||||||
|
@ -884,7 +884,6 @@ tech-sub-sec h2 {
|
||||||
|
|
||||||
.price-calc-section .card .description span {
|
.price-calc-section .card .description span {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
margin-left: 4px;
|
|
||||||
margin-left: 0px;
|
margin-left: 0px;
|
||||||
width: 30%;
|
width: 30%;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
|
@ -1002,7 +1001,7 @@ tech-sub-sec h2 {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media screen and (min-width: 480px) and (max-width: 767px) {
|
@media (min-width: 576px) and (max-width: 767px) {
|
||||||
.logo-wrap, .logo-wrap-1 {
|
.logo-wrap, .logo-wrap-1 {
|
||||||
width: 50%;
|
width: 50%;
|
||||||
padding: 15px 30px !important;
|
padding: 15px 30px !important;
|
||||||
|
@ -1010,7 +1009,7 @@ tech-sub-sec h2 {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media(max-width:990px) {
|
@media(max-width:991px) {
|
||||||
.pricing-section .text {
|
.pricing-section .text {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
margin-bottom: 40px;
|
margin-bottom: 40px;
|
||||||
|
@ -1174,7 +1173,7 @@ tech-sub-sec h2 {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@media(max-width:540px) {
|
@media(max-width:575px) {
|
||||||
.logo-wrap {
|
.logo-wrap {
|
||||||
padding: 30px;
|
padding: 30px;
|
||||||
}
|
}
|
||||||
|
@ -1201,7 +1200,6 @@ tech-sub-sec h2 {
|
||||||
}
|
}
|
||||||
.price-calc-section .card .description span {
|
.price-calc-section .card .description span {
|
||||||
font-size: 15px;
|
font-size: 15px;
|
||||||
margin-left: 0px;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
from django.conf.urls import url
|
from django.conf.urls import url
|
||||||
|
|
||||||
from .views import (
|
from .views import (
|
||||||
IndexView, BetaProgramView, LandingProgramView, BetaAccessView,
|
# BetaProgramView, SuccessView, LandingProgramView,
|
||||||
SuccessView, PaymentOrderView, OrderConfirmationView,
|
IndexView, BetaAccessView, PaymentOrderView, OrderConfirmationView,
|
||||||
WhyDataCenterLightView, ContactUsView
|
WhyDataCenterLightView, ContactUsView
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -25,7 +25,6 @@
|
||||||
{% trans "Generate" %}
|
{% trans "Generate" %}
|
||||||
</button>
|
</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<div>
|
||||||
<h2>{% trans "Using existing key" %}</h2>
|
<h2>{% trans "Using existing key" %}</h2>
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
<div>
|
<div>
|
||||||
<div class="virtual-machine-container dashboard-container">
|
<div class="virtual-machine-container dashboard-container">
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-md-9 col-md-offset-2">
|
<div class="col-md-8 col-md-offset-2">
|
||||||
<form method="POST" action="" novalidate class="form-ssh">
|
<form method="POST" action="" novalidate class="form-ssh">
|
||||||
{% csrf_token %}
|
{% csrf_token %}
|
||||||
<div class="page-header">
|
<div class="page-header">
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import csv
|
import csv
|
||||||
import logging
|
import logging
|
||||||
import os
|
import os
|
||||||
|
import pprint
|
||||||
import re
|
import re
|
||||||
from collections import Counter, OrderedDict
|
from collections import Counter, OrderedDict
|
||||||
from itertools import zip_longest
|
from itertools import zip_longest
|
||||||
|
@ -17,8 +18,11 @@ RE_PATTERNS = {
|
||||||
'view_html': '[\'\"](.*\.html)',
|
'view_html': '[\'\"](.*\.html)',
|
||||||
'html_html': '{% (?:extends|include) [\'\"]?(.*\.html)',
|
'html_html': '{% (?:extends|include) [\'\"]?(.*\.html)',
|
||||||
'html_style': '{% static [\'\"]?(.*\.css)',
|
'html_style': '{% static [\'\"]?(.*\.css)',
|
||||||
|
'css_media': (
|
||||||
|
'^\s*\@media([^{]+)\{\s*([\s\S]*?})\s*}'
|
||||||
|
),
|
||||||
'css_selector': (
|
'css_selector': (
|
||||||
'^\s*([.#\[:_A-Za-z][.#\[\]\(\)=:+~\-_A-Za-z0-9\s>,]*)'
|
'^\s*([.#\[:_A-Za-z][^{]*)'
|
||||||
'{([\s\S]*?)}'
|
'{([\s\S]*?)}'
|
||||||
),
|
),
|
||||||
'html_class': 'class=[\'\"]([a-zA-Z0-9-_\s]*)',
|
'html_class': 'class=[\'\"]([a-zA-Z0-9-_\s]*)',
|
||||||
|
@ -54,123 +58,176 @@ class Command(BaseCommand):
|
||||||
for app in apps_list:
|
for app in apps_list:
|
||||||
if options['css']:
|
if options['css']:
|
||||||
self.optimize_css(app)
|
self.optimize_css(app)
|
||||||
else:
|
# else:
|
||||||
self.optimize_all(app)
|
# optimize_all(app)
|
||||||
|
|
||||||
def optimize_css(self, app_name):
|
def optimize_css(self, app_name):
|
||||||
# get html and css files used in the app
|
# get html and css files used in the app
|
||||||
files = self.get_files(app_name)
|
files = get_files(app_name)
|
||||||
# get_selectors_from_css
|
# get_selectors_from_css
|
||||||
css_selectors = self.get_selectors_css(files['style'])
|
css_selectors = get_selectors_css(files['style'])
|
||||||
# get_selectors_from_html
|
# get_selectors_from_html
|
||||||
html_selectors = self.get_selectors_html(files['html'])
|
html_selectors = get_selectors_html(files['html'])
|
||||||
|
# get duplication of css rules from css files
|
||||||
|
css_dup_report = get_css_duplication(css_selectors)
|
||||||
|
|
||||||
# duplicate css selectors in stylesheets
|
|
||||||
for file, selectors in css_selectors.items():
|
|
||||||
count = {}
|
|
||||||
for selector in selectors:
|
|
||||||
if selector[0] in count:
|
|
||||||
count[selector[0]] += 1
|
|
||||||
print(file, selector[0], count[selector[0]])
|
|
||||||
else:
|
|
||||||
count[selector[0]] = 1
|
|
||||||
|
|
||||||
def get_files(self, app_name):
|
def get_files(app_name):
|
||||||
# the view file for the app
|
# the view file for the app
|
||||||
app_view = os.path.join(settings.PROJECT_DIR, app_name, 'views.py')
|
app_view = os.path.join(settings.PROJECT_DIR, app_name, 'views.py')
|
||||||
# get template files called from the view
|
# get template files called from the view
|
||||||
all_html_list = file_match_pattern(app_view, ['view_html'])[0]
|
all_html_list = file_match_pattern(app_view, 'view_html')
|
||||||
# list of unique template files
|
# list of unique template files
|
||||||
uniq_html_list = list(OrderedDict.fromkeys(all_html_list).keys())
|
uniq_html_list = list(OrderedDict.fromkeys(all_html_list).keys())
|
||||||
# list of stylesheets
|
# list of stylesheets
|
||||||
all_style_list = []
|
all_style_list = []
|
||||||
file_patterns = ['html_html', 'html_style']
|
file_patterns = ['html_html', 'html_style']
|
||||||
# get html and css files called from within templates
|
# get html and css files called from within templates
|
||||||
i = 0
|
i = 0
|
||||||
while i < len(uniq_html_list):
|
while i < len(uniq_html_list):
|
||||||
template_name = uniq_html_list[i]
|
template_name = uniq_html_list[i]
|
||||||
try:
|
try:
|
||||||
# a dict containing 'html' and 'css' files
|
# a dict containing 'html' and 'css' files
|
||||||
temp_files = templates_match_pattern(
|
temp_files = templates_match_pattern(
|
||||||
template_name, file_patterns
|
template_name, file_patterns
|
||||||
)
|
)
|
||||||
except template.exceptions.TemplateDoesNotExist as e:
|
except template.exceptions.TemplateDoesNotExist as e:
|
||||||
print("template file not found: ", str(e))
|
print("template file not found: ", str(e))
|
||||||
all_html_list = [
|
all_html_list = [
|
||||||
h for h in all_html_list if h != template_name
|
h for h in all_html_list if h != template_name
|
||||||
]
|
]
|
||||||
del uniq_html_list[i]
|
del uniq_html_list[i]
|
||||||
else:
|
else:
|
||||||
all_html_list.extend(temp_files[0])
|
all_html_list.extend(temp_files[0])
|
||||||
uniq_html_list = list(
|
uniq_html_list = list(
|
||||||
OrderedDict.fromkeys(all_html_list).keys()
|
OrderedDict.fromkeys(all_html_list).keys()
|
||||||
)
|
)
|
||||||
all_style_list.extend(temp_files[1])
|
all_style_list.extend(temp_files[1])
|
||||||
i += 1
|
i += 1
|
||||||
# counter dict for the html files called from view
|
# counter dict for the html files called from view
|
||||||
result = {
|
result = {
|
||||||
'html': Counter(all_html_list),
|
'html': Counter(all_html_list),
|
||||||
'style': Counter(all_style_list)
|
'style': Counter(all_style_list)
|
||||||
}
|
}
|
||||||
print(result)
|
print(result)
|
||||||
return result
|
return result
|
||||||
|
|
||||||
def get_selectors_css(self, files):
|
|
||||||
selectors = {}
|
|
||||||
for file in files:
|
|
||||||
if any(vendor in file for vendor in ['bootstrap', 'font-awesome']):
|
|
||||||
continue
|
|
||||||
result = finders.find(file)
|
|
||||||
if result:
|
|
||||||
selectors[file] = file_match_pattern(
|
|
||||||
result, ['css_selector']
|
|
||||||
)[0]
|
|
||||||
return selectors
|
|
||||||
|
|
||||||
def get_selectors_html(self, files):
|
def get_selectors_css(files):
|
||||||
selectors = {}
|
selectors = {}
|
||||||
for file in files:
|
media_selectors = {}
|
||||||
results = templates_match_pattern(file, ['html_class', 'html_id'])
|
for file in files:
|
||||||
|
if any(vendor in file for vendor in ['bootstrap', 'font-awesome']):
|
||||||
|
continue
|
||||||
|
result = finders.find(file)
|
||||||
|
if result:
|
||||||
|
with open(result) as f:
|
||||||
|
data = f.read()
|
||||||
|
media_selectors[file] = string_match_pattern(
|
||||||
|
data, 'css_media'
|
||||||
|
)
|
||||||
|
new_data = string_replace_pattern(
|
||||||
|
data, 'css_media'
|
||||||
|
)
|
||||||
selectors[file] = {
|
selectors[file] = {
|
||||||
'class': results[0],
|
'default': string_match_pattern(
|
||||||
'id': results[0],
|
new_data, 'css_selector'
|
||||||
|
)
|
||||||
}
|
}
|
||||||
return selectors
|
# pp = pprint.PrettyPrinter(compact=False, width=120)
|
||||||
|
# pp.pprint(media_selectors)
|
||||||
|
|
||||||
def selectors_css(self, results, filename='frontend'):
|
for file, match_list in media_selectors.items():
|
||||||
full_filename = '../optimize_' + filename + '.csv'
|
for match in match_list:
|
||||||
output_file = os.path.join(
|
query = match[0]
|
||||||
settings.PROJECT_DIR, full_filename
|
block_text = ' '.join(match[1].split())
|
||||||
)
|
results = string_match_pattern(
|
||||||
with open(output_file, 'w', newline='') as f:
|
block_text, 'css_selector'
|
||||||
w = csv.writer(f)
|
)
|
||||||
print(zip_longest(*results))
|
f_query = ' '.join(query.replace(':', ': ').split())
|
||||||
for r in zip_longest(*results):
|
if f_query in selectors[file]:
|
||||||
w.writerow(r)
|
selectors[file][f_query].extend(results)
|
||||||
|
else:
|
||||||
|
selectors[file][f_query] = results
|
||||||
|
# pp.pprint(selectors)
|
||||||
|
return selectors
|
||||||
|
|
||||||
|
|
||||||
|
def get_selectors_html(files):
|
||||||
|
selectors = {}
|
||||||
|
for file in files:
|
||||||
|
results = templates_match_pattern(file, ['html_class', 'html_id'])
|
||||||
|
selectors[file] = {
|
||||||
|
'class': results[0],
|
||||||
|
'id': results[1],
|
||||||
|
}
|
||||||
|
return selectors
|
||||||
|
|
||||||
|
|
||||||
def file_match_pattern(file, patterns):
|
def file_match_pattern(file, patterns):
|
||||||
results = []
|
|
||||||
with open(file) as f:
|
with open(file) as f:
|
||||||
data = f.read()
|
data = f.read()
|
||||||
for p in patterns:
|
results = string_match_pattern(data, patterns)
|
||||||
results.append(
|
|
||||||
re.findall(re.compile(RE_PATTERNS[p], re.MULTILINE), data)
|
|
||||||
)
|
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def string_match_pattern(data, patterns):
|
||||||
|
if not isinstance(patterns, str):
|
||||||
|
results = []
|
||||||
|
for p in patterns:
|
||||||
|
re_pattern = re.compile(RE_PATTERNS[p], re.MULTILINE)
|
||||||
|
results.append(re.findall(re_pattern, data))
|
||||||
|
else:
|
||||||
|
re_pattern = re.compile(RE_PATTERNS[patterns], re.MULTILINE)
|
||||||
|
results = re.findall(re_pattern, data)
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def string_replace_pattern(data, patterns):
|
||||||
|
if not isinstance(patterns, str):
|
||||||
|
for p in patterns:
|
||||||
|
re_pattern = re.compile(RE_PATTERNS[p], re.MULTILINE)
|
||||||
|
data = re.sub(re_pattern, '', data)
|
||||||
|
else:
|
||||||
|
re_pattern = re.compile(RE_PATTERNS[patterns], re.MULTILINE)
|
||||||
|
data = re.sub(re_pattern, '', data)
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
def templates_match_pattern(template_name, patterns):
|
def templates_match_pattern(template_name, patterns):
|
||||||
t = template.loader.get_template(template_name)
|
t = template.loader.get_template(template_name)
|
||||||
data = t.template.source
|
data = t.template.source
|
||||||
results = []
|
results = string_match_pattern(data, patterns)
|
||||||
for p in patterns:
|
|
||||||
results.append(
|
|
||||||
re.findall(re.compile(RE_PATTERNS[p], re.MULTILINE), data)
|
|
||||||
)
|
|
||||||
return results
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def get_css_duplication(css_selectors):
|
||||||
|
# duplicate css selectors in stylesheets
|
||||||
|
for file in css_selectors:
|
||||||
|
print(file)
|
||||||
|
for media in css_selectors[file]:
|
||||||
|
print(' '.join(media.replace(':', ': ').split()))
|
||||||
|
print(len(css_selectors[file][media]), 'rules')
|
||||||
|
# for selector in selectors:
|
||||||
|
# if selector[0] in count:
|
||||||
|
# count[selector[0]] += 1
|
||||||
|
# # print(file, selector[0], count[selector[0]])
|
||||||
|
# else:
|
||||||
|
# count[selector[0]] = 1
|
||||||
|
|
||||||
|
|
||||||
|
def write_report(results, filename='frontend'):
|
||||||
|
full_filename = '../optimize_' + filename + '.csv'
|
||||||
|
output_file = os.path.join(
|
||||||
|
settings.PROJECT_DIR, full_filename
|
||||||
|
)
|
||||||
|
with open(output_file, 'w', newline='') as f:
|
||||||
|
w = csv.writer(f)
|
||||||
|
print(zip_longest(*results))
|
||||||
|
for r in zip_longest(*results):
|
||||||
|
w.writerow(r)
|
||||||
|
|
||||||
|
|
||||||
html_tags = [
|
html_tags = [
|
||||||
"a",
|
"a",
|
||||||
"abbr",
|
"abbr",
|
||||||
|
@ -280,6 +337,6 @@ html_tags = [
|
||||||
"wbr"
|
"wbr"
|
||||||
]
|
]
|
||||||
|
|
||||||
exempt_classes = [
|
bootstrap_classes = [
|
||||||
"active",
|
"active",
|
||||||
]
|
]
|
||||||
|
|
Loading…
Reference in a new issue