Update
This commit is contained in:
parent
d8b6e05efb
commit
51dc008aa7
|
@ -4,7 +4,6 @@ import Options from '../../options';
|
||||||
import { StolenData } from './stolen-data';
|
import { StolenData } from './stolen-data';
|
||||||
import { getshorthost, useEmitter } from '../../util';
|
import { getshorthost, useEmitter } from '../../util';
|
||||||
import { getMemory } from '../../memory';
|
import { getMemory } from '../../memory';
|
||||||
browser.browserAction.setBadgeBackgroundColor({ color: '#ffb900' });
|
|
||||||
|
|
||||||
async function getCurrentTab() {
|
async function getCurrentTab() {
|
||||||
const [tab] = await browser.tabs.query({
|
const [tab] = await browser.tabs.query({
|
||||||
|
@ -19,7 +18,10 @@ import './../../styles/global.scss';
|
||||||
import './sidebar.scss';
|
import './sidebar.scss';
|
||||||
|
|
||||||
const Sidebar = () => {
|
const Sidebar = () => {
|
||||||
const [origin, setOrigin] = React.useState<string | null>(null);
|
// const [origin, setOrigin] = React.useState<string | null>(null);
|
||||||
|
const url = new URL(document.location.toString());
|
||||||
|
const origin = url.searchParams.get('origin');
|
||||||
|
|
||||||
const [minValueLength, setMinValueLength] = React.useState<number | null>(
|
const [minValueLength, setMinValueLength] = React.useState<number | null>(
|
||||||
localStorage.getItem('minValueLength') === null
|
localStorage.getItem('minValueLength') === null
|
||||||
? 7
|
? 7
|
||||||
|
@ -45,23 +47,7 @@ const Sidebar = () => {
|
||||||
: false
|
: false
|
||||||
);
|
);
|
||||||
|
|
||||||
React.useEffect(() => {
|
|
||||||
const listener = async () => {
|
|
||||||
console.log('tab change!');
|
|
||||||
const tab = await getCurrentTab();
|
|
||||||
const url = new URL(tab.url);
|
|
||||||
if (url.origin.startsWith('moz-extension')) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setOrigin(url.origin);
|
|
||||||
};
|
|
||||||
|
|
||||||
browser.tabs.onUpdated.addListener(listener);
|
|
||||||
listener();
|
|
||||||
return () => {
|
|
||||||
browser.tabs.onUpdated.removeListener(listener);
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
for (const cluster of Object.values(getMemory().getClustersForOrigin(origin))) {
|
for (const cluster of Object.values(getMemory().getClustersForOrigin(origin))) {
|
||||||
|
@ -71,7 +57,7 @@ const Sidebar = () => {
|
||||||
}
|
}
|
||||||
|
|
||||||
return setMarksOccurrence(false);
|
return setMarksOccurrence(false);
|
||||||
}, [eventCounts['*'], origin]);
|
}, [eventCounts['*']]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="sidebar">
|
<div className="sidebar">
|
||||||
|
|
|
@ -1,11 +1,16 @@
|
||||||
@import '../../styles/colors.scss';
|
@import '../../styles/colors.scss';
|
||||||
|
|
||||||
|
body {
|
||||||
|
width: 400px;
|
||||||
|
overflow-x: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
.toolbar {
|
.toolbar {
|
||||||
padding: 0 0.75rem;
|
padding: 0.125rem 1rem;
|
||||||
|
|
||||||
.header {
|
.header {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 2.75rem 1fr 1.25rem;
|
grid-template-columns: 1.75rem 1fr 1.25rem;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
max-height: 3.5rem;
|
max-height: 3.5rem;
|
||||||
min-height: 3.5rem;
|
min-height: 3.5rem;
|
||||||
|
@ -21,13 +26,13 @@
|
||||||
flex-flow: column;
|
flex-flow: column;
|
||||||
flex-wrap: nowrap;
|
flex-wrap: nowrap;
|
||||||
font-size: 0.875rem;
|
font-size: 0.875rem;
|
||||||
font-weight: 600;
|
font-weight: 700;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
padding-left: 1rem;
|
padding-left: 1rem;
|
||||||
color: #000;
|
color: #000;
|
||||||
|
|
||||||
&--hyperlink {
|
&--hyperlink {
|
||||||
font-weight: 400;
|
font-weight: 600;
|
||||||
color: $ultra-black-color;
|
color: $ultra-black-color;
|
||||||
max-height: 2rem;
|
max-height: 2rem;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
@ -44,43 +49,125 @@
|
||||||
|
|
||||||
.summary {
|
.summary {
|
||||||
display: flex;
|
display: flex;
|
||||||
.counters{
|
flex-flow: column;
|
||||||
|
justify-content: center;
|
||||||
|
padding-bottom: 1.5rem;
|
||||||
|
border-bottom: 1px solid $light-grey;
|
||||||
|
|
||||||
|
.counters-wrapper {
|
||||||
|
flex-flow: row;
|
||||||
display: flex;
|
display: flex;
|
||||||
.counter{
|
justify-content: center;
|
||||||
font-size: 1rem;
|
|
||||||
font-weight: 900;
|
.counters {
|
||||||
|
display: flex;
|
||||||
|
flex-flow: column;
|
||||||
|
align-content: flex-start;
|
||||||
|
justify-content: center;
|
||||||
|
margin-right: 1rem;
|
||||||
|
|
||||||
|
.counter {
|
||||||
|
font-size: 1.25rem;
|
||||||
|
font-weight: 700;
|
||||||
|
align-items: center;
|
||||||
|
display: flex;
|
||||||
|
|
||||||
|
img {
|
||||||
|
margin-right: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
span {
|
||||||
|
margin-right: 2rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:nth-child(1) {
|
||||||
|
margin-bottom: 1rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.big-counter {
|
||||||
|
font-size: 6rem;
|
||||||
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.big-counter {
|
|
||||||
font-size: 6rem;
|
|
||||||
font-size: 900;
|
|
||||||
|
|
||||||
|
.notice {
|
||||||
|
font-size: 0.875rem;
|
||||||
|
font-weight: 700;
|
||||||
|
text-align: center;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.warning-container {
|
.details {
|
||||||
background-color: $pale-yellow;
|
padding-top: 1.5rem;
|
||||||
border-left: 4px solid $contrast-yellow;
|
|
||||||
margin-top: 0.5rem;
|
|
||||||
font-size: 0.875rem;
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: calc(100% - 2rem) 2rem;
|
|
||||||
align-items: flex-start;
|
|
||||||
|
|
||||||
span {
|
p {
|
||||||
padding: 1rem;
|
font-size: 0.875rem;
|
||||||
|
padding-bottom: 1rem;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.about {
|
||||||
|
p {
|
||||||
|
font-size: 0.875rem;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.actions {
|
||||||
|
padding: 2rem 0;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
a {
|
||||||
|
font-size: 0.875rem;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #000;
|
||||||
|
margin-right: 0.5rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
button {
|
.button {
|
||||||
justify-content: flex-end;
|
border: 0;
|
||||||
border: none;
|
outline: 0;
|
||||||
|
height: 3rem;
|
||||||
|
font-size: 0.875rem;
|
||||||
|
line-height: 0.875rem;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
color: $ultra-black-color;
|
|
||||||
line-height: 1.25rem;
|
&--details {
|
||||||
background: transparent;
|
margin-right: 0.5rem;
|
||||||
padding: 0.5rem 0.5rem;
|
color: #000;
|
||||||
display: flex;
|
text-decoration:underline;
|
||||||
align-self: flex-start;
|
background-color: #fff;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
|
||||||
|
&--report {
|
||||||
|
font-weight: 800;
|
||||||
|
padding: 0 1.5rem;
|
||||||
|
margin-left: 0.5rem;
|
||||||
|
background-color: #000;
|
||||||
|
color: $icd-rentgen-color;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
background-image: linear-gradient(
|
||||||
|
to right,
|
||||||
|
$icd-rentgen-color 0%,
|
||||||
|
$icd-rentgen-color 4%,
|
||||||
|
#000 4%,
|
||||||
|
#000 100%
|
||||||
|
);
|
||||||
|
animation: slidebg 1s cubic-bezier(0.19, 1, 0.22, 1) infinite;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes slidebg {
|
||||||
|
to {
|
||||||
|
background-position: 155px;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
import React from 'react';
|
import React, { Fragment, ReactElement } from 'react';
|
||||||
import ReactDOM from 'react-dom';
|
import ReactDOM from 'react-dom';
|
||||||
import { useEmitter } from '../../util';
|
import { useEmitter } from '../../util';
|
||||||
import { getMemory } from '../../memory';
|
import { getMemory } from '../../memory';
|
||||||
browser.browserAction.setBadgeBackgroundColor({ color: '#ffb900' });
|
|
||||||
|
|
||||||
async function getCurrentTab() {
|
async function getCurrentTab() {
|
||||||
const [tab] = await browser.tabs.query({
|
const [tab] = await browser.tabs.query({
|
||||||
|
@ -17,34 +16,15 @@ import './toolbar.scss';
|
||||||
|
|
||||||
const Toolbar = () => {
|
const Toolbar = () => {
|
||||||
const [origin, setOrigin] = React.useState<string | null>(null);
|
const [origin, setOrigin] = React.useState<string | null>(null);
|
||||||
const [minValueLength, setMinValueLength] = React.useState<number | null>(
|
|
||||||
localStorage.getItem('minValueLength') === null
|
|
||||||
? 7
|
|
||||||
: (localStorage.getItem('minValueLength') as unknown as number)
|
|
||||||
);
|
|
||||||
const [cookiesOnly, setCookiesOnly] = React.useState<boolean>(false);
|
|
||||||
const [stolenDataView, setStolenDataView] = React.useState<boolean>(true);
|
const [stolenDataView, setStolenDataView] = React.useState<boolean>(true);
|
||||||
const [cookiesOrOriginOnly, setCookiesOrOriginOnly] = React.useState<boolean>(false);
|
|
||||||
const [eventCounts, setEventCounts] = useEmitter(getMemory());
|
const [eventCounts, setEventCounts] = useEmitter(getMemory());
|
||||||
const [marksOccurrence, setMarksOccurrence] = React.useState<boolean>(false);
|
const [cookieDomainCopy, setCookieDomainCopy] = React.useState<string | null>(null);
|
||||||
const [warningDataDialogAck, setWarningDataDialogAck] = React.useState<boolean>(
|
const [exposedOriginDomainCopy, setExposedOriginDomainCopy] = React.useState<string | null>(
|
||||||
localStorage.getItem('warningDataDialogAck') === null
|
null
|
||||||
? true
|
|
||||||
: localStorage.getItem('warningDataDialogAck') == 'true'
|
|
||||||
? true
|
|
||||||
: false
|
|
||||||
);
|
|
||||||
const [logoVisibility, setLogoVisibility] = React.useState<boolean>(
|
|
||||||
localStorage.getItem('logoVisibility') === null
|
|
||||||
? false
|
|
||||||
: localStorage.getItem('logoVisibility') == 'true'
|
|
||||||
? true
|
|
||||||
: false
|
|
||||||
);
|
);
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
const listener = async () => {
|
const listener = async () => {
|
||||||
console.log('tab change!');
|
|
||||||
const tab = await getCurrentTab();
|
const tab = await getCurrentTab();
|
||||||
const url = new URL(tab.url);
|
const url = new URL(tab.url);
|
||||||
if (url.origin.startsWith('moz-extension')) {
|
if (url.origin.startsWith('moz-extension')) {
|
||||||
|
@ -61,26 +41,74 @@ const Toolbar = () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
React.useEffect(() => {
|
React.useEffect(() => {
|
||||||
for (const cluster of Object.values(getMemory().getClustersForOrigin(origin))) {
|
const exposedOriginDomains = Object.values(getMemory().getClustersForOrigin(origin))
|
||||||
if (cluster.hasMarks()) {
|
.filter((cluster) => cluster.exposesOrigin())
|
||||||
return setMarksOccurrence(true);
|
.map((cluster) => cluster.id);
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return setMarksOccurrence(false);
|
setExposedOriginDomainCopy('');
|
||||||
|
const first_sentence = `Strona ${origin} wysłała informacje o części Twojej historii przeglądania do `;
|
||||||
|
|
||||||
|
switch (exposedOriginDomains.length) {
|
||||||
|
case 0:
|
||||||
|
return null;
|
||||||
|
case 1:
|
||||||
|
return setExposedOriginDomainCopy(first_sentence + `${exposedOriginDomains[0]}.`);
|
||||||
|
case 2:
|
||||||
|
return setExposedOriginDomainCopy(
|
||||||
|
first_sentence + `${exposedOriginDomains[0]} oraz ${exposedOriginDomains[1]}.`
|
||||||
|
);
|
||||||
|
case 3:
|
||||||
|
return setExposedOriginDomainCopy(
|
||||||
|
first_sentence +
|
||||||
|
`${exposedOriginDomains[0]}, ${exposedOriginDomains[1]} oraz ${exposedOriginDomains[2]}.`
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
return setExposedOriginDomainCopy(
|
||||||
|
first_sentence +
|
||||||
|
`${exposedOriginDomains[0]}, ${exposedOriginDomains[1]} (i ${
|
||||||
|
exposedOriginDomains.length - 2 < 2
|
||||||
|
? 2
|
||||||
|
: exposedOriginDomains.length - 2
|
||||||
|
} innych).`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}, [eventCounts['*'], origin]);
|
||||||
|
|
||||||
|
React.useEffect(() => {
|
||||||
|
const cookieDomains = Object.values(getMemory().getClustersForOrigin(origin))
|
||||||
|
.filter((cluster) => cluster.hasCookies())
|
||||||
|
.map((cluster) => cluster.id);
|
||||||
|
const first_sentence = `Strona ${origin} dokonała zapisu i odczytu plików Cookie dla domen `;
|
||||||
|
|
||||||
|
switch (cookieDomains.length) {
|
||||||
|
case 0:
|
||||||
|
return null;
|
||||||
|
case 1:
|
||||||
|
return setCookieDomainCopy(first_sentence + `${cookieDomains[0]}.`);
|
||||||
|
case 2:
|
||||||
|
return setCookieDomainCopy(
|
||||||
|
first_sentence + `${cookieDomains[0]} oraz ${cookieDomains[1]}.`
|
||||||
|
);
|
||||||
|
case 3:
|
||||||
|
return setCookieDomainCopy(
|
||||||
|
first_sentence +
|
||||||
|
`${cookieDomains[0]}, ${cookieDomains[1]} oraz ${cookieDomains[2]}.`
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
return setCookieDomainCopy(
|
||||||
|
first_sentence +
|
||||||
|
`${cookieDomains[0]}, ${cookieDomains[1]} (i ${
|
||||||
|
cookieDomains.length - 2 < 2 ? 2 : cookieDomains.length - 2
|
||||||
|
} innych).`
|
||||||
|
);
|
||||||
|
}
|
||||||
}, [eventCounts['*'], origin]);
|
}, [eventCounts['*'], origin]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="toolbar">
|
<div className="toolbar">
|
||||||
<header className="header">
|
<header className="header">
|
||||||
<img src="../../assets/icon-addon.svg" height={32}></img>
|
<img src="../../assets/icon-addon.svg" height={32}></img>
|
||||||
<div
|
<div className="webpage-metadata">
|
||||||
className={
|
|
||||||
logoVisibility
|
|
||||||
? 'webpage-metadata'
|
|
||||||
: 'webpage-metadata webpage-metadata--without-logo'
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{origin ? (
|
{origin ? (
|
||||||
<>
|
<>
|
||||||
<span>Analiza strony</span>
|
<span>Analiza strony</span>
|
||||||
|
@ -102,39 +130,72 @@ const Toolbar = () => {
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<section className="summary">
|
<section className="summary">
|
||||||
<div>
|
<div className="counters-wrapper">
|
||||||
<div className="counters">
|
<div className="counters">
|
||||||
<div className="counter counter--browser-history">12</div>
|
<div className="counter counter--browser-history">
|
||||||
<div className="counter counter--cookies">21</div>
|
<img src="/assets/icons/warning.svg" width="24" height="24" />
|
||||||
|
<span data-event={`${eventCounts['*']}`}>
|
||||||
|
{
|
||||||
|
Object.values(getMemory().getClustersForOrigin(origin)).filter(
|
||||||
|
(cluster) => cluster.exposesOrigin()
|
||||||
|
).length
|
||||||
|
}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="counter counter--cookies">
|
||||||
|
<img src="/assets/icons/cookie.svg" width="24" height="24" />
|
||||||
|
<span data-event={`${eventCounts['*']}`}>
|
||||||
|
{
|
||||||
|
Object.values(getMemory().getClustersForOrigin(origin)).filter(
|
||||||
|
(cluster) => cluster.hasCookies()
|
||||||
|
).length
|
||||||
|
}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="big-counter" data-event={`${eventCounts['*']}`}>
|
||||||
|
{Object.values(getMemory().getClustersForOrigin(origin)).length}
|
||||||
</div>
|
</div>
|
||||||
<div className="big-counter">33</div>
|
|
||||||
</div>
|
</div>
|
||||||
<p>Liczba wykrytych domen podmiotów trzecich</p>
|
<span className="notice">Liczba wykrytych domen podmiotów trzecich</span>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section className="details">
|
<section className="details">
|
||||||
<p>
|
<p>{exposedOriginDomainCopy}</p>
|
||||||
Strona wp.pl wysłała informacje o części Twojej historii przeglądania do
|
<p>{cookieDomainCopy}</p>
|
||||||
facebook.com, adnsx.com (i 43 innych).
|
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
Dokonała też zapisu i odczytu plików Cookie dla domen doubleclick.google.net,
|
|
||||||
3dsio.com (i 59 innych).
|
|
||||||
</p>
|
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
<section className="warning-container">
|
{exposedOriginDomainCopy !== null || cookieDomainCopy !== null ? (
|
||||||
<span>
|
<Fragment>
|
||||||
<strong>Takie przetwarzanie danych może być niezgodne z prawem.</strong> Kliknij
|
<section className="about">
|
||||||
w przycisk <i>Generuj raport</i>, aby pomóc ustalić, czy ta strona nie narusza
|
<p>
|
||||||
RODO.
|
Takie przetwarzanie danych może być niezgodne z prawem. Kliknij w
|
||||||
</span>
|
przycisk „Generuj raport”, aby pomóc ustalić, czy ta strona nie narusza
|
||||||
</section>
|
RODO.
|
||||||
|
</p>
|
||||||
<section className="actions">
|
</section>
|
||||||
<a href="">Pokaż szczegóły</a>
|
<section className="actions">
|
||||||
<button>Generuj raport</button>
|
<button
|
||||||
</section>
|
className="button button--details"
|
||||||
|
onClick={() => {
|
||||||
|
const params = [
|
||||||
|
'height=' + screen.height,
|
||||||
|
'width=' + screen.width,
|
||||||
|
'fullscreen=yes',
|
||||||
|
].join(',');
|
||||||
|
window.open(
|
||||||
|
`/components/sidebar/sidebar.html?origin=${origin}`,
|
||||||
|
'new_window',
|
||||||
|
params
|
||||||
|
);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Pokaż szczegóły
|
||||||
|
</button>
|
||||||
|
<button className="button button--report">Generuj raport</button>
|
||||||
|
</section>
|
||||||
|
</Fragment>
|
||||||
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
|
@ -6,7 +6,7 @@ import { RequestCluster } from './request-cluster';
|
||||||
function setDomainsNumber(counter: number, tabId: number) {
|
function setDomainsNumber(counter: number, tabId: number) {
|
||||||
browser.browserAction.setBadgeText({ text: counter.toString(), tabId });
|
browser.browserAction.setBadgeText({ text: counter.toString(), tabId });
|
||||||
browser.browserAction.setTitle({
|
browser.browserAction.setTitle({
|
||||||
title: `Rentgen - liczba wykrytych podmiotów: ${counter}`,
|
title: 'Rentgen',
|
||||||
tabId,
|
tabId,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,12 @@ export default class Memory extends EventEmitter {
|
||||||
this.origin_to_history[request.origin][shorthost].add(request);
|
this.origin_to_history[request.origin][shorthost].add(request);
|
||||||
this.emit('change', false, shorthost, 'registered request(shorthost emit)');
|
this.emit('change', false, shorthost, 'registered request(shorthost emit)');
|
||||||
|
|
||||||
console.log(this.getClustersForOrigin(request.origin));
|
Object.values(this.getClustersForOrigin(request.origin)).some((cluster) =>
|
||||||
|
cluster.hasCookies()
|
||||||
|
)
|
||||||
|
? browser.browserAction.setBadgeBackgroundColor({ color: '#ff726b' })
|
||||||
|
: browser.browserAction.setBadgeBackgroundColor({ color: '#ffb900' });
|
||||||
|
|
||||||
setDomainsNumber(
|
setDomainsNumber(
|
||||||
Object.values(this.getClustersForOrigin(request.origin)).length,
|
Object.values(this.getClustersForOrigin(request.origin)).length,
|
||||||
request.tabId
|
request.tabId
|
||||||
|
|
|
@ -10,11 +10,7 @@
|
||||||
user-select: none;
|
user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
body {
|
|
||||||
width: 400px;
|
|
||||||
height: 600px;
|
|
||||||
overflow-x: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
html {
|
html {
|
||||||
font-size: 1rem;
|
font-size: 1rem;
|
||||||
|
|
Loading…
Reference in New Issue
Block a user