├── logo.png
├── assets
├── icon.png
├── iconLarge.png
└── preview.gif
├── generated-icon.png
├── webfonts
└── fa-solid-900.woff2
├── test
└── data
│ └── 05-versions-space.pdf
├── models
└── README.md
├── .replit
├── LICENSE
├── .gitignore
├── package.json
├── main.js
├── README.md
├── index.html
├── styles.css
├── fileProcessor.js
├── renderer.js
└── all.min.css
/logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/AgenticA5/A5-PII-Anonymizer/HEAD/logo.png
--------------------------------------------------------------------------------
/assets/icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/AgenticA5/A5-PII-Anonymizer/HEAD/assets/icon.png
--------------------------------------------------------------------------------
/assets/iconLarge.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/AgenticA5/A5-PII-Anonymizer/HEAD/assets/iconLarge.png
--------------------------------------------------------------------------------
/assets/preview.gif:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/AgenticA5/A5-PII-Anonymizer/HEAD/assets/preview.gif
--------------------------------------------------------------------------------
/generated-icon.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/AgenticA5/A5-PII-Anonymizer/HEAD/generated-icon.png
--------------------------------------------------------------------------------
/webfonts/fa-solid-900.woff2:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/AgenticA5/A5-PII-Anonymizer/HEAD/webfonts/fa-solid-900.woff2
--------------------------------------------------------------------------------
/test/data/05-versions-space.pdf:
--------------------------------------------------------------------------------
1 | %PDF-1.4
2 | 1 0 obj
3 | << /Type /Catalog /Pages 2 0 R >>
4 | endobj
5 | 2 0 obj
6 | << /Type /Pages /Kids [3 0 R] /Count 1 >>
7 | endobj
8 | 3 0 obj
9 | << /Type /Page /Parent 2 0 R /MediaBox [0 0 612 792] >>
10 | endobj
11 | xref
12 | 0 4
13 | 0000000000 65535 f
14 | 0000000009 00000 n
15 | 0000000053 00000 n
16 | 0000000100 00000 n
17 | trailer
18 | << /Size 4 /Root 1 0 R >>
19 | startxref
20 | 165
21 | %%EOF
22 |
--------------------------------------------------------------------------------
/models/README.md:
--------------------------------------------------------------------------------
1 | Your model config and ONNX model go here. In this case protectai/lakshyakh93-deberta_finetuned_pii-onnx. It's too large for github unfortunately and requires another github repo here. There is an onnx folder that needs a `model_quantized.onnx` file inside of an onnx folder... This will be your largest file and the file we cannot included here on Github due to size limitations.
2 |
3 | Find more on this model here: https://huggingface.co/iiiorg/piiranha-v1-detect-personal-information
4 |
5 | and here:
6 |
7 | https://huggingface.co/iiiorg/piiranha-v1-detect-personal-information/discussions/8
8 |
9 |
10 |
--------------------------------------------------------------------------------
/.replit:
--------------------------------------------------------------------------------
1 | modules = ["nodejs-20"]
2 |
3 | [nix]
4 | channel = "stable-24_05"
5 |
6 | [workflows]
7 | runButton = "Project"
8 |
9 | [[workflows.workflow]]
10 | name = "Project"
11 | mode = "parallel"
12 | author = "agent"
13 |
14 | [[workflows.workflow.tasks]]
15 | task = "workflow.run"
16 | args = "Electron App"
17 |
18 | [[workflows.workflow]]
19 | name = "Electron App"
20 | author = "agent"
21 |
22 | [workflows.workflow.metadata]
23 | agentRequireRestartOnSave = false
24 |
25 | [[workflows.workflow.tasks]]
26 | task = "packager.installForAll"
27 |
28 | [[workflows.workflow.tasks]]
29 | task = "shell.exec"
30 | args = "electron ."
31 |
32 | [deployment]
33 | run = ["sh", "-c", "electron ."]
34 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | MIT License
2 |
3 | Copyright (c) 2025 A5
4 |
5 | Permission is hereby granted, free of charge, to any person obtaining a copy
6 | of this software and associated documentation files (the "Software"), to deal
7 | in the Software without restriction, including without limitation the rights
8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9 | copies of the Software, and to permit persons to whom the Software is
10 | furnished to do so, subject to the following conditions:
11 |
12 | The above copyright notice and this permission notice shall be included in all
13 | copies or substantial portions of the Software.
14 |
15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 | SOFTWARE.
22 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Node modules and package manager artifacts
2 | /node_modules/
3 | /dist/
4 | .pnp/
5 | .pnp.js
6 |
7 | # Optional npm cache directory
8 | .npm
9 |
10 | # Logs
11 | logs
12 | *.log
13 | npm-debug.log*
14 | yarn-debug.log*
15 | yarn-error.log*
16 |
17 | # Runtime data
18 | pids
19 | *.pid
20 | *.seed
21 | *.pid.lock
22 |
23 | # Directory for instrumented libs generated by jscoverage/JSCover
24 | lib-cov
25 |
26 | # Coverage directory used by tools like istanbul
27 | coverage/
28 |
29 | # Production build output
30 | build/
31 | # Note: dist/ is intentionally kept
32 |
33 | # Dependency directories
34 | jspm_packages/
35 |
36 | # Optional REPL history
37 | .node_repl_history
38 |
39 | # Output of 'npm pack'
40 | *.tgz
41 |
42 | # Yarn Integrity file
43 | .yarn-integrity
44 |
45 | # dotenv environment variables file (ignore .env files in any folder)
46 | **/.env
47 |
48 | # OS generated files
49 | .DS_Store
50 | Thumbs.db
51 |
52 | #example files
53 | oldExampleFileProcessor.js
54 | oldExampleMain.js
55 | oldExampleStyles.css
56 |
57 | # IDE or Editor directories and files
58 | .vscode/
59 | .idea/
60 | *.suo
61 | *.ntvs*
62 | *.njsproj
63 | *.sln
64 |
65 | # Miscellaneous
66 | *.bak
67 | *.tmp
68 |
69 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "type": "module",
3 | "name": "a5-pii-anonymizer",
4 | "version": "0.0.1",
5 | "main": "main.js",
6 | "description": "A short description of your app",
7 | "scripts": {
8 | "test": "echo \"Error: no test specified\" && exit 1",
9 | "dev": "electron .",
10 | "build": "electron-builder",
11 | "build:mac": "electron-builder --mac",
12 | "build:win": "electron-builder --win",
13 | "build:linux": "electron-builder --linux"
14 | },
15 | "keywords": [],
16 | "author": "",
17 | "license": "ISC",
18 | "dependencies": {
19 | "exceljs": "^4.3.0",
20 | "@xenova/transformers": "2.17.2",
21 | "mammoth": "^1.5.2",
22 | "docx": "^8.4.0",
23 | "pdf-parse": "^1.1.1",
24 | "pdf-lib": "^1.17.1"
25 | },
26 | "devDependencies": {
27 | "electron": "^34.2.0",
28 | "electron-builder": "^23.6.0"
29 | },
30 | "build": {
31 | "productName": "A5 PII Anonymizer",
32 | "appId": "com.a5.piiAnonymizer",
33 | "mac": {
34 | "icon": "assets/icon",
35 | "category": "public.app-category.utilities"
36 | },
37 | "win": {
38 | "icon": "assets/icon"
39 | },
40 | "directories": {
41 | "buildResources": "assets"
42 | },
43 | "files": [
44 | "**/*",
45 | {
46 | "from": "assets",
47 | "to": "assets",
48 | "filter": [
49 | "icon.png"
50 | ]
51 | }
52 | ]
53 | }
54 | }
55 |
--------------------------------------------------------------------------------
/main.js:
--------------------------------------------------------------------------------
1 | import { app, BrowserWindow, ipcMain, dialog, shell } from 'electron';
2 | import path from 'path';
3 | import fs from 'fs';
4 | import { FileProcessor } from './fileProcessor.js';
5 | import { fileURLToPath } from 'url';
6 |
7 | let isLLMInitialized = false; // track if LLM is loaded once
8 |
9 | const __filename = fileURLToPath(import.meta.url);
10 | const __dirname = path.dirname(__filename);
11 |
12 | let mainWindow;
13 |
14 | function createWindow() {
15 | mainWindow = new BrowserWindow({
16 | width: 900,
17 | height: 600,
18 | webPreferences: {
19 | nodeIntegration: true,
20 | contextIsolation: false,
21 | },
22 | backgroundColor: '#1a1a1a',
23 | });
24 | // mainWindow.webContents.openDevTools(); // uncomment if you want the console
25 |
26 | mainWindow.loadFile('index.html');
27 | }
28 |
29 | app.whenReady().then(() => {
30 | createWindow();
31 |
32 | // macOS Dock icon
33 | if (process.platform === 'darwin') {
34 | if (app.isPackaged) {
35 | // In production, icon is inside resources/assets/icon.png
36 | const iconPath = path.join(process.resourcesPath, 'assets', 'icon.png');
37 | app.dock.setIcon(iconPath);
38 | } else {
39 | // In dev mode, use a local path
40 | const devIconPath = path.join(__dirname, 'assets', 'icon.png');
41 | app.dock.setIcon(devIconPath);
42 | }
43 | }
44 | });
45 |
46 | app.on('window-all-closed', () => {
47 | if (process.platform !== 'darwin') app.quit();
48 | });
49 |
50 | app.on('activate', () => {
51 | if (BrowserWindow.getAllWindows().length === 0) createWindow();
52 | });
53 |
54 | ipcMain.handle('select-output-directory', async () => {
55 | const result = await dialog.showOpenDialog(mainWindow, {
56 | properties: ['openDirectory'],
57 | });
58 | if (!result.canceled && result.filePaths.length > 0) {
59 | return result.filePaths[0];
60 | }
61 | return null;
62 | });
63 |
64 | ipcMain.handle('select-input-directory', async () => {
65 | const result = await dialog.showOpenDialog(mainWindow, {
66 | properties: ['openDirectory'],
67 | });
68 | if (!result.canceled && result.filePaths.length > 0) {
69 | return result.filePaths[0];
70 | }
71 | return null;
72 | });
73 |
74 | ipcMain.handle('process-file', async (event, { filePath, outputDir }) => {
75 | try {
76 | const fileName = path.basename(filePath);
77 |
78 | // If LLM not yet loaded, notify the renderer
79 | if (!isLLMInitialized) {
80 | mainWindow.webContents.send('log-message', "Initializing LLM (first-time load)...");
81 | }
82 |
83 | mainWindow.webContents.send('log-message', `Processing: ${fileName}`);
84 |
85 | const directory = outputDir || path.dirname(filePath);
86 | const newFileName = FileProcessor.generateOutputFileName(fileName);
87 | const outputPath = path.join(directory, newFileName);
88 |
89 | await FileProcessor.processFile(filePath, outputPath);
90 |
91 | // Mark LLM as initialized after first file
92 | isLLMInitialized = true;
93 |
94 | mainWindow.webContents.send('log-message', `Finished: ${fileName}`);
95 | return { success: true, outputPath };
96 | } catch (error) {
97 | console.error("Error in process-file IPC:", error);
98 | mainWindow.webContents.send('log-message', `Error: ${error.message}`);
99 | return { success: false, error: error.message };
100 | }
101 | });
102 |
103 | // Open a folder or URL
104 | ipcMain.handle('open-folder', async (event, folderPath) => {
105 | if (folderPath) {
106 | // If it looks like a URL (starts with http or https), open in external browser
107 | if (folderPath.startsWith('http')) {
108 | shell.openExternal(folderPath);
109 | } else {
110 | // Otherwise, treat as a local file path
111 | shell.openPath(folderPath);
112 | }
113 | }
114 | });
115 |
--------------------------------------------------------------------------------
/README.md:
--------------------------------------------------------------------------------
1 | # A5 PII Anonymizer
2 |
3 | ### Built-In LLM Desktop App Preview:
4 |
5 |
6 | This repository provides an **Electron** desktop application for **locally anonymizing documents** before sending them to advanced Large Language Models (LLMs). By stripping out personal or identifiable information using a **context-aware** model, you can safely train or query external LLMs (e.g., OpenAI’s o3 model) with minimal privacy risk.
7 |
8 | ## Motivation
9 |
10 | - **PII Removal**: Traditional RegEx-based anonymization often fails on nuanced data. With an ONNX-based model, you gain context-aware detection for **names, addresses, phone numbers, etc.**
11 | - **Safe LLM Usage**: Many companies need to keep real customer or employee data **internal** but still want to leverage powerful external LLMs. This tool helps them do so by anonymizing data **on their end** first.
12 | - **Flexible**: Supports `.txt`, `.docx`, `.xls(x)`, `.csv`, `.pdf`, and more. Converts text, merges tokens, and replaces them with consistent pseudonyms.
13 |
14 | ## Key Features
15 |
16 | 1. **Electron App**: Cross-platform desktop UI built on HTML/CSS/JS.
17 | 2. **Daily Limit**: **100** documents per day for the free tier (by default). You can raise or remove it if you prefer—this is open source.
18 | 3. **Context-Aware**: Relies on a local ONNX model downloaded separately (due to GitHub’s file-size constraints).
19 | 4. **Mapping** (Pro Mode): If you enable Pro, the app can produce a JSON file mapping each original entity (e.g., “John Smith”) to its anonymized token (e.g., “NAME_1”).
20 | 5. **MIT License**: Free to modify and distribute. We welcome contributions.
21 |
22 | ## Getting Started
23 |
24 | 1. **Clone or Download** this repository.
25 | - Model available here: [https://huggingface.co/iiiorg/piiranha-v1-detect-personal-information/tree/main](https://huggingface.co/iiiorg/piiranha-v1-detect-personal-information/tree/main)
26 | - Download the ONNX model (`.onnx` files) from our external link (not included here due to size constraints).
27 | - Place it under `./models/protectai/lakshyakh93-deberta_finetuned_pii-onnx/` or as directed in `fileProcessor.js`.
28 | 3. **Install Dependencies**:
29 | ```bash
30 | npm install
31 | ```
32 | 4. **Run (Dev Mode)**:
33 | ```bash
34 | npm run dev
35 | ```
36 | Or if you prefer:
37 | ```bash
38 | npx electron .
39 | ```
40 | 5. **Build / Package** (macOS example):
41 | ```bash
42 | npm run build:mac
43 |
44 |
45 | ### Basic Usage
46 |
47 | - **Drop or Select Files**: The main UI allows you to drag-and-drop or pick multiple files/folders.
48 | - **Output Directory**: Choose where the anonymized files should be placed.
49 | - **Anonymize**: Click “Anonymize Files” to run.
50 | - **Mapping (Pro)**: If you have a Pro key, the app will create an additional `-map.json` file capturing each replaced entity.
51 |
52 | ## How It Works
53 |
54 | - **Electron**:
55 | - **`main.js`**: Spawns the main window, handles file selection, passes tasks to `FileProcessor`.
56 | - **`renderer.js`**: Manages the UI (index.html), user interactions, daily usage counters, and “Pro” logic.
57 | - **`fileProcessor.js`**:
58 | - Loads the local ONNX model (via `@xenova/transformers`).
59 | - Identifies personal data by context (names, addresses, etc.).
60 | - Replaces them with tokens (`NAME_1`, `PHONE_NUMBER_3`, etc.).
61 | - If Pro, writes a JSON mapping for re-identification.
62 | - **Local Model**:
63 | - We rely on a context-aware token classification model. This is significantly more effective than simple RegEx for real-world PII.
64 |
65 | ## Limitations & Notes
66 |
67 | - **Daily 100-File Limit**: By default, the free version only processes 100 documents per day. This is purely enforced in the UI. Since it’s open source, you can remove or change it as needed.
68 | - **No Guarantee**: Even context-aware models can miss certain edge cases. Always manually review if 100% privacy is critical.
69 | - **Cross-Platform**:
70 | - macOS builds are tested on both M-series (ARM) and Intel.
71 | - Windows and Linux builds are also available.
72 |
73 | ## Contributing
74 |
75 | We use the **MIT License**, so feel free to open pull requests, modify the code, or adapt it for your needs. If you make improvements or fix bugs, please share them back!
76 |
77 | ---
78 |
79 | **Thanks for checking out the A5 PII Anonymizer.** We hope this helps you safely leverage powerful LLMs with real data while keeping personal information private.
--------------------------------------------------------------------------------
/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 | A5 PII Anonymizer
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
15 |
16 | What's Included in Pro?
17 |
18 |
19 |
20 |
21 |
22 |
23 | PII Anonymizer
24 |
25 |
26 |
27 |
28 |
29 |
Drag & Drop files here or click to select
30 |
37 |
38 |
39 |
40 |
41 |
44 |
45 |
46 |
47 |
48 |
49 |
52 |
53 |
54 |
55 |
66 |
67 |
68 |
71 |
72 |
73 |
76 |
77 |
78 |
79 |
80 |
81 |
85 |
86 |
87 |
90 |
91 |
92 |
93 |
94 |
95 |
×
96 |
Upgrade to Pro
97 |
98 |
Device ID:
99 |
100 |
101 |
102 |
103 |
104 |
Upgrade to Pro:
105 |
106 |
107 |
108 |
109 |
Already have a key?
110 |
111 |
112 |
113 |
114 |
115 |
116 |
117 |
118 |
119 |
120 |
121 |
×
122 |
Pro Features
123 |
124 | - Lift the daily 100-file limit
125 | - Access an entity map for re-identification
126 |
127 |
128 |
See full details at amicus5.com/store/pa
129 |
130 |
131 |
132 |
133 |
134 |
135 |
×
136 |
Manage Plan
137 |
You are currently on the Pro plan.
138 |
139 |
140 | Visit store for more info.
141 |
142 |
143 |
144 |
145 |
146 |
147 |
148 |
--------------------------------------------------------------------------------
/styles.css:
--------------------------------------------------------------------------------
1 | :root {
2 | --bg-primary: #1a1a1a;
3 | --bg-secondary: #2d2d2d;
4 | --text-primary: #ffffff;
5 | --text-secondary: #b3b3b3;
6 | --accent: #007bff;
7 | --success: #28a745;
8 | --error: #dc3545;
9 | }
10 |
11 | body {
12 | margin: 0;
13 | padding: 20px;
14 | font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Arial, sans-serif;
15 | background-color: var(--bg-primary);
16 | color: var(--text-primary);
17 | position: relative;
18 | }
19 |
20 | .container {
21 | max-width: 800px;
22 | margin: 0 auto;
23 | position: relative;
24 | }
25 |
26 | h1 {
27 | text-align: center;
28 | margin-bottom: 30px;
29 | }
30 |
31 | .company-logo {
32 | vertical-align: middle;
33 | margin-right: 5px;
34 | height: 36px; /* or whatever size you want */
35 | position: relative;
36 | top: -3px; /* lift the image 2px above the baseline */
37 | }
38 |
39 |
40 | .drop-zone {
41 | border: 2px dashed var(--text-secondary);
42 | border-radius: 8px;
43 | padding: 40px;
44 | text-align: center;
45 | cursor: pointer;
46 | transition: all 0.3s ease;
47 | }
48 |
49 | .drop-zone:hover {
50 | border-color: var(--accent);
51 | background-color: rgba(0, 123, 255, 0.1);
52 | }
53 |
54 | .drop-zone i {
55 | font-size: 48px;
56 | margin-bottom: 15px;
57 | color: var(--text-secondary);
58 | }
59 |
60 | .file-list {
61 | margin: 20px 0;
62 | padding: 15px;
63 | background-color: var(--bg-secondary);
64 | border-radius: 6px;
65 | }
66 |
67 | .file-list ul {
68 | list-style: none;
69 | padding: 0;
70 | margin: 0;
71 | }
72 |
73 | .file-list li {
74 | padding: 5px 0;
75 | border-bottom: 1px solid var(--bg-primary);
76 | }
77 |
78 | .file-list li:last-child {
79 | border-bottom: none;
80 | }
81 |
82 | .settings-section {
83 | margin: 20px 0;
84 | }
85 |
86 | .input-group {
87 | margin-bottom: 15px;
88 | }
89 |
90 | .input-group label {
91 | display: block;
92 | margin-bottom: 5px;
93 | color: var(--text-secondary);
94 | }
95 |
96 | input[type="text"] {
97 | width: 100%;
98 | padding: 8px;
99 | border: 1px solid var(--bg-secondary);
100 | border-radius: 4px;
101 | background-color: var(--bg-secondary);
102 | color: var(--text-primary);
103 | }
104 |
105 | .output-dir-group {
106 | display: flex;
107 | gap: 10px;
108 | }
109 |
110 | .output-dir-group input {
111 | flex: 1;
112 | }
113 |
114 | .button {
115 | padding: 8px 16px;
116 | border: none;
117 | border-radius: 4px;
118 | cursor: pointer;
119 | display: inline-flex;
120 | align-items: center;
121 | gap: 8px;
122 | transition: all 0.3s ease;
123 | }
124 |
125 | .button i {
126 | font-size: 14px;
127 | }
128 |
129 | .button.primary {
130 | background-color: var(--accent);
131 | color: white;
132 | }
133 |
134 | .button.secondary {
135 | background-color: var(--bg-secondary);
136 | color: var(--text-primary);
137 | }
138 |
139 | .button:disabled {
140 | opacity: 0.6;
141 | cursor: not-allowed;
142 | }
143 |
144 | .progress {
145 | margin: 20px 0;
146 | background-color: var(--bg-secondary);
147 | border-radius: 4px;
148 | overflow: hidden;
149 | }
150 |
151 | .progress-bar {
152 | height: 4px;
153 | background-color: var(--accent);
154 | width: 0%;
155 | transition: width 0.3s ease;
156 | }
157 |
158 | .status {
159 | margin-top: 20px;
160 | padding: 10px;
161 | border-radius: 4px;
162 | text-align: center;
163 | }
164 |
165 | .status.success {
166 | background-color: var(--success);
167 | }
168 |
169 | .status.error {
170 | background-color: var(--error);
171 | }
172 |
173 | .hidden {
174 | display: none;
175 | }
176 |
177 | .output-link {
178 | margin-top: 20px;
179 | text-align: center;
180 | }
181 |
182 | .output-link a {
183 | color: var(--accent);
184 | text-decoration: none;
185 | font-weight: bold;
186 | }
187 |
188 | /* PRO container top-right */
189 | .pro-container {
190 | position: absolute;
191 | top: 20px;
192 | right: 20px;
193 | text-align: right;
194 | z-index: 9998; /* ensure on top of drop-zone */
195 | }
196 |
197 | .pro-upgrade {
198 | background-color: #fff;
199 | color: #000;
200 | cursor: pointer;
201 | }
202 | .pro-upgrade:hover {
203 | opacity: 0.8;
204 | }
205 |
206 | .pro-active {
207 | background-color: #666;
208 | color: #fff;
209 | cursor: default;
210 | }
211 |
212 | /* smaller text, no underline, pointer */
213 | .pro-info-link {
214 | display: block;
215 | font-size: 0.8rem;
216 | color: #fff;
217 | margin-top: 4px;
218 | cursor: pointer;
219 | text-decoration: none;
220 | }
221 | .pro-info-link:hover {
222 | opacity: 0.8;
223 | }
224 |
225 | /* modal overlay */
226 | .modal {
227 | position: fixed;
228 | z-index: 9999; /* above pro-container */
229 | left: 0;
230 | top: 0;
231 | width: 100%;
232 | height: 100%;
233 | overflow: auto;
234 | background-color: rgba(0, 0, 0, 0.7);
235 | display: none;
236 | justify-content: center;
237 | align-items: center;
238 | }
239 |
240 | .modal.show {
241 | display: flex;
242 | }
243 |
244 | .modal-content {
245 | background-color: var(--bg-secondary);
246 | padding: 20px;
247 | border-radius: 8px;
248 | max-width: 500px;
249 | width: 90%;
250 | color: var(--text-primary);
251 | position: relative;
252 | }
253 |
254 | .modal-close {
255 | position: absolute;
256 | top: 10px;
257 | right: 10px;
258 | cursor: pointer;
259 | font-size: 24px;
260 | color: var(--text-primary);
261 | }
262 |
263 | .device-id-container {
264 | display: flex;
265 | align-items: center;
266 | gap: 8px;
267 | }
268 |
269 | /* Force 1px white border around these fields */
270 | .device-id-field,
271 | .pro-key-input {
272 | border: 1px solid #fff !important;
273 | }
274 |
275 | /* store button with wide style */
276 | .store-button {
277 | background-color: #fff;
278 | color: #000;
279 | font-size: 0.9rem;
280 | }
281 | .store-button:hover {
282 | opacity: 0.8;
283 | }
284 | .wide-button {
285 | width: 100%;
286 | justify-content: center;
287 | }
288 |
289 | .white-border-btn {
290 | background-color: var(--bg-secondary);
291 | border: 1px solid #fff;
292 | color: #fff;
293 | }
294 | .white-border-btn:hover {
295 | opacity: 0.8;
296 | }
297 |
298 | /* pro-key-input with white border, slightly bigger radius */
299 | .pro-key-input {
300 | width: 100%;
301 | background-color: #333;
302 | border-radius: 6px;
303 | color: #fff;
304 | margin-bottom: 10px;
305 | padding: 8px;
306 | }
307 |
308 | /* store-link styling */
309 | .store-link {
310 | color: #fff;
311 | text-decoration: underline;
312 | cursor: pointer;
313 | }
314 | .store-link:hover {
315 | opacity: 0.8;
316 | }
317 |
318 | /* Manage Plan modal's error button for "Downgrade" */
319 | .error-btn {
320 | background-color: #dc3545;
321 | color: #fff;
322 | }
323 | .error-btn:hover {
324 | opacity: 0.8;
325 | }
326 |
327 | /* center the wide button */
328 | .centered-button {
329 | text-align: center;
330 | margin: 10px 0;
331 | }
332 |
333 | /* message area inside the upgrade modal */
334 | .key-message {
335 | margin-top: 10px;
336 | padding: 8px;
337 | border-radius: 4px;
338 | text-align: center;
339 | font-size: 0.9rem;
340 | }
341 |
--------------------------------------------------------------------------------
/fileProcessor.js:
--------------------------------------------------------------------------------
1 | import fs from 'fs';
2 | import path from 'path';
3 | import ExcelJS from 'exceljs';
4 | import mammoth from 'mammoth';
5 | import { Document, Packer, Paragraph } from 'docx';
6 | import pdfParse from 'pdf-parse';
7 | import { PDFDocument } from 'pdf-lib';
8 |
9 | import { pipeline, env } from '@xenova/transformers';
10 | import { fileURLToPath } from 'url';
11 |
12 | // ES module paths
13 | const __filename = fileURLToPath(import.meta.url);
14 | const __dirname = path.dirname(__filename);
15 |
16 | // Transformers.js environment
17 | env.localModelPath = path.join(__dirname, 'models');
18 | env.allowRemoteModels = false;
19 | env.quantized = false;
20 |
21 | // Toggle whether we use LLM-based anonymization
22 | const useLLM = true;
23 |
24 | // Pipeline reference
25 | let nerPipeline = null;
26 |
27 | // Pseudonym counters/mappings
28 | const pseudonymCounters = {};
29 | const pseudonymMapping = {};
30 |
31 | /**
32 | * Returns a consistent pseudonym for a given entity text + type.
33 | */
34 | function getPseudonym(entityText, entityType) {
35 | if (pseudonymMapping[entityText]) {
36 | return pseudonymMapping[entityText];
37 | }
38 | if (!pseudonymCounters[entityType]) {
39 | pseudonymCounters[entityType] = 1;
40 | }
41 | const pseudonym = `${entityType}_${pseudonymCounters[entityType]++}`;
42 | pseudonymMapping[entityText] = pseudonym;
43 | return pseudonym;
44 | }
45 |
46 | /**
47 | * Aggressively merges consecutive tokens of the same entity type,
48 | * removing whitespace/punctuation from each token, then concatenating.
49 | * e.g. “Bay,” + “ona,” + “Wil” + “ber” => “BayonaWilber”
50 | */
51 | function aggressiveMergeTokens(predictions) {
52 | if (!predictions || predictions.length === 0) return [];
53 |
54 | const merged = [];
55 | let current = null;
56 |
57 | for (const pred of predictions) {
58 | const type = pred.entity.replace(/^(B-|I-)/, '');
59 | // Remove whitespace/punctuation from each token
60 | let word = pred.word.replace(/\s+/g, '').replace(/[^\w\s.,'-]/g, '');
61 | word = word.trim();
62 | if (!word) continue;
63 |
64 | if (!current) {
65 | current = { type, text: word };
66 | } else if (current.type === type) {
67 | // Same entity => unify
68 | current.text += word;
69 | } else {
70 | // Different entity => push old one, start new
71 | merged.push(current);
72 | current = { type, text: word };
73 | }
74 | }
75 | if (current) {
76 | merged.push(current);
77 | }
78 | return merged;
79 | }
80 |
81 | /**
82 | * Safely escapes all regex meta-characters in a string.
83 | */
84 | function escapeRegexChars(str) {
85 | return str.replace(/[-/\\^$*+?.()|[\]{}]/g, '\\$&');
86 | }
87 |
88 | /**
89 | * Builds a fuzzy regex (with 'g' + 'i') that matches the merged string ignoring spacing/punctuation.
90 | */
91 | function buildFuzzyRegex(mergedString) {
92 | // Remove punctuation from mergedString
93 | let noPunc = mergedString.replace(/[^\w]/g, '');
94 | if (!noPunc) {
95 | return null;
96 | }
97 |
98 | // Escape special regex chars
99 | noPunc = escapeRegexChars(noPunc);
100 |
101 | // Build a pattern that allows any non-alphanumeric between letters
102 | let pattern = '';
103 | for (const char of noPunc) {
104 | pattern += `${char}[^a-zA-Z0-9]*`;
105 | }
106 | // No trailing slice, to avoid bracket issues.
107 |
108 | if (!pattern) {
109 | return null;
110 | }
111 |
112 | try {
113 | return new RegExp(pattern, 'ig');
114 | } catch (err) {
115 | console.warn(`Regex build failed for pattern="${pattern}". Error: ${err.message}`);
116 | return null;
117 | }
118 | }
119 |
120 | /**
121 | * Loads the PII detection model from local files, if not already loaded.
122 | */
123 | async function loadNERModel() {
124 | if (!nerPipeline) {
125 | console.log("Loading PII detection model from local files...");
126 | nerPipeline = await pipeline('token-classification', 'protectai/lakshyakh93-deberta_finetuned_pii-onnx');
127 | console.log("Model loaded.");
128 | }
129 | return nerPipeline;
130 | }
131 |
132 | /**
133 | * The main anonymization function.
134 | * 1) Runs the pipeline
135 | * 2) Merges partial tokens
136 | * 3) Uses a fuzzy global regex to replace each merged token with a pseudonym
137 | */
138 | async function anonymizeText(text) {
139 | let processedText = String(text);
140 |
141 | const ner = await loadNERModel();
142 | console.log("Internal LLM processing...");
143 | const predictions = await ner(processedText);
144 | console.log("Raw predicted tokens:", predictions);
145 |
146 | const merged = aggressiveMergeTokens(predictions);
147 | console.log("Aggressively merged tokens:", merged);
148 |
149 | for (const obj of merged) {
150 | const entityType = obj.type;
151 | const mergedString = obj.text;
152 | if (!mergedString) continue;
153 |
154 | const pseudonym = getPseudonym(mergedString, entityType);
155 | const fuzzyRegex = buildFuzzyRegex(mergedString);
156 | if (!fuzzyRegex) {
157 | console.log(`Skipping zero-length or invalid pattern for mergedString="${mergedString}"`);
158 | continue;
159 | }
160 |
161 | console.log(`Replacing fuzzy match of "${mergedString}" => regex ${fuzzyRegex} with "${pseudonym}"`);
162 |
163 | // Single-pass global replace
164 | processedText = processedText.replace(fuzzyRegex, pseudonym);
165 | }
166 |
167 | console.log("LLM processing complete.");
168 | return processedText;
169 | }
170 |
171 | export class FileProcessor {
172 | static async processFile(filePath, outputPath) {
173 | return new Promise(async (resolve, reject) => {
174 | try {
175 | const ext = path.extname(filePath).toLowerCase();
176 | console.log(`Processing file: ${filePath}`);
177 |
178 | if (ext === '.txt' || ext === '.csv') {
179 | // Text-based approach
180 | console.log(`Processing text file: ${filePath}`);
181 | const content = fs.readFileSync(filePath, 'utf8');
182 | let newContent;
183 | if (useLLM) {
184 | console.log("LLM anonymization enabled. Processing text...");
185 | const anonymizedText = await anonymizeText(content);
186 | newContent = "Anonymized\n\n" + anonymizedText;
187 | } else {
188 | console.log("LLM anonymization disabled. Using default processing.");
189 | newContent = "Anonymized\n\n" + content;
190 | }
191 | fs.writeFileSync(outputPath, newContent, 'utf8');
192 | console.log(`Text file processed and saved to: ${outputPath}`);
193 | resolve(true);
194 |
195 | } else if (ext === '.xlsx') {
196 | // Excel partial coverage
197 | console.log(`Processing Excel file: ${filePath}`);
198 | const workbook = new ExcelJS.Workbook();
199 | await workbook.xlsx.readFile(filePath);
200 |
201 | for (const worksheet of workbook.worksheets) {
202 | for (let i = 1; i <= worksheet.rowCount; i++) {
203 | const row = worksheet.getRow(i);
204 | for (let j = 1; j <= row.cellCount; j++) {
205 | const cell = row.getCell(j);
206 | if (typeof cell.value === 'string') {
207 | console.log(`Anonymizing cell [Row ${i}, Col ${j}] with value: ${cell.value}`);
208 | cell.value = await anonymizeText(cell.value);
209 | }
210 | }
211 | }
212 | }
213 |
214 | await workbook.xlsx.writeFile(outputPath);
215 | console.log(`Excel file processed and saved to: ${outputPath}`);
216 | resolve(true);
217 |
218 | } else if (ext === '.docx') {
219 | // DOCX: mammoth + docx approach
220 | console.log(`Processing DOCX file: ${filePath}`);
221 | const { value: docxText } = await mammoth.extractRawText({ path: filePath });
222 | console.log("Extracted DOCX text:", docxText);
223 |
224 | let anonymizedDocxText = docxText;
225 | if (useLLM) {
226 | anonymizedDocxText = await anonymizeText(docxText);
227 | }
228 |
229 | // Create minimal docx with 'docx' library
230 | const doc = new Document({
231 | sections: [
232 | {
233 | children: [ new Paragraph(anonymizedDocxText) ],
234 | },
235 | ],
236 | });
237 | const buffer = await Packer.toBuffer(doc);
238 | fs.writeFileSync(outputPath, buffer);
239 | console.log(`DOCX file processed and saved to: ${outputPath}`);
240 | resolve(true);
241 |
242 | } else if (ext === '.pdf') {
243 | // PDF: pdf-parse + pdf-lib approach
244 | console.log(`Processing PDF file: ${filePath}`);
245 | const dataBuffer = fs.readFileSync(filePath);
246 | const data = await pdfParse(dataBuffer);
247 | const pdfText = data.text;
248 | console.log("Extracted PDF text:", pdfText);
249 |
250 | let anonymizedPdfText = pdfText;
251 | if (useLLM) {
252 | anonymizedPdfText = await anonymizeText(pdfText);
253 | }
254 |
255 | // Create a minimal PDF with pdf-lib
256 | const doc = await PDFDocument.create();
257 | const page = doc.addPage();
258 | page.drawText(anonymizedPdfText, { x: 50, y: 700, size: 12 });
259 | const pdfBytes = await doc.save();
260 | fs.writeFileSync(outputPath, pdfBytes);
261 | console.log(`PDF file processed and saved to: ${outputPath}`);
262 | resolve(true);
263 |
264 | } else {
265 | // For other file types, just copy
266 | console.log(`Processing binary file: ${filePath}`);
267 | fs.copyFileSync(filePath, outputPath);
268 | console.log(`Binary file copied to: ${outputPath}`);
269 | resolve(true);
270 | }
271 | } catch (error) {
272 | console.error("Error in processFile:", error);
273 | reject(error);
274 | }
275 | });
276 | }
277 |
278 | static generateOutputFileName(originalName) {
279 | const ext = path.extname(originalName);
280 | const baseName = path.basename(originalName, ext);
281 | return `${baseName}-anon${ext}`;
282 | }
283 |
284 | static validateFileType(filePath) {
285 | const supportedTypes = [
286 | '.doc', '.docx', '.xls', '.xlsx', '.csv', '.pdf', '.txt'
287 | ];
288 | const ext = path.extname(filePath).toLowerCase();
289 | return supportedTypes.includes(ext);
290 | }
291 | }
292 |
--------------------------------------------------------------------------------
/renderer.js:
--------------------------------------------------------------------------------
1 | /***** renderer.js *****/
2 | const { ipcRenderer } = require('electron');
3 | const fs = require('fs');
4 | const path = require('path');
5 | const os = require('os');
6 |
7 | // Global userState
8 | let userState = {
9 | deviceID: null,
10 | isPro: false,
11 | dailyCount: 0,
12 | dailyDate: null
13 | };
14 | let autoUpdate= false;
15 |
16 | // Simple hash for your local testing
17 | function simpleHash(str) {
18 | let hash = 0;
19 | for (let i = 0; i < str.length; i++) {
20 | hash = (hash << 5) - hash + str.charCodeAt(i);
21 | hash |= 0;
22 | }
23 | return hash.toString(16);
24 | }
25 |
26 | // DOM elements
27 | let selectedFiles = [];
28 | let outputDirectory = null;
29 | let lastDialogTime = 0;
30 |
31 | const fileInput = document.getElementById('file-input');
32 | const dropZone = document.getElementById('drop-zone');
33 | const selectFolderBtn = document.getElementById('select-folder');
34 | const fileListDiv = document.getElementById('file-list');
35 | const filesUl = document.getElementById('files-ul');
36 | const clearFilesBtn = document.getElementById('clear-files');
37 | const outputDirInput = document.getElementById('output-dir');
38 | const selectOutputBtn = document.getElementById('select-output');
39 | const processButton = document.getElementById('process-button');
40 | const progress = document.getElementById('progress');
41 | const progressBar = document.querySelector('.progress-bar');
42 | const statusDiv = document.getElementById('status');
43 | const outputLinkDiv = document.getElementById('output-link');
44 | const openOutputFolderLink = document.getElementById('open-output-folder');
45 |
46 | // Logs area
47 | const logArea = document.getElementById('log-area');
48 | const logMessages = document.getElementById('log-messages');
49 |
50 | // Pro container
51 | const proContainer = document.getElementById('pro-container');
52 | const proButton = document.getElementById('pro-button');
53 | const proInfoLink = document.getElementById('pro-info-link');
54 |
55 | // Upgrade modal
56 | const upgradeModal = document.getElementById('upgrade-modal');
57 | const upgradeClose = document.getElementById('upgrade-close');
58 | const deviceIdField = document.getElementById('device-id-field');
59 | const copyDeviceIdBtn = document.getElementById('copy-device-id');
60 | const proKeyInput = document.getElementById('pro-key-input');
61 | const validateKeyBtn = document.getElementById('validate-key-button');
62 | const upgradeStoreBtn = document.getElementById('upgrade-store-btn');
63 | const keyMessageDiv = document.getElementById('key-message'); // for invalid/success messages
64 |
65 | // Info modal
66 | const infoModal = document.getElementById('info-modal');
67 | const infoClose = document.getElementById('info-close');
68 |
69 | // Manage Plan modal
70 | const manageModal = document.getElementById('manage-modal');
71 | const manageClose = document.getElementById('manage-close');
72 | const downgradeBtn = document.getElementById('downgrade-btn');
73 |
74 | // On load: restore output dir
75 | const storedOutput = localStorage.getItem('outputDirectory');
76 | if (storedOutput) {
77 | outputDirectory = storedOutput;
78 | outputDirInput.value = storedOutput;
79 | outputLinkDiv.classList.remove('hidden');
80 | openOutputFolderLink.onclick = async () => {
81 | await ipcRenderer.invoke('open-folder', outputDirectory);
82 | };
83 | }
84 |
85 | // Load userState
86 | function loadUserState() {
87 | const saved = localStorage.getItem('userState');
88 | if (saved) {
89 | userState = JSON.parse(saved);
90 | } else {
91 | userState = {
92 | deviceID: null,
93 | isPro: false,
94 | dailyCount: 0,
95 | dailyDate: null
96 | };
97 | }
98 | if (!userState.deviceID) {
99 | userState.deviceID = generateDeviceID(10);
100 | saveUserState();
101 | }
102 | checkDailyReset();
103 | }
104 |
105 | function saveUserState() {
106 | localStorage.setItem('userState', JSON.stringify(userState));
107 | }
108 |
109 | function generateDeviceID(length) {
110 | const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
111 | let result = '';
112 | for (let i = 0; i < length; i++) {
113 | result += chars.charAt(Math.floor(Math.random() * chars.length));
114 | }
115 | return result;
116 | }
117 |
118 | function checkDailyReset() {
119 | const today = getLocalDateString();
120 | if (userState.dailyDate !== today) {
121 | userState.dailyDate = today;
122 | userState.dailyCount = 0;
123 | saveUserState();
124 | }
125 | }
126 |
127 | function getLocalDateString() {
128 | const now = new Date();
129 | return now.toLocaleDateString('en-CA');
130 | }
131 |
132 | loadUserState();
133 | updateProUI();
134 |
135 | dropZone.addEventListener('click', () => {
136 | const now = Date.now();
137 | if (now - lastDialogTime < 1000) return;
138 | lastDialogTime = now;
139 | fileInput.value = "";
140 | fileInput.click();
141 | });
142 |
143 | fileInput.addEventListener('change', async (e) => {
144 | await handleInputItems(e.target.files);
145 | });
146 |
147 | dropZone.addEventListener('dragover', (e) => {
148 | e.preventDefault();
149 | dropZone.style.borderColor = 'var(--accent)';
150 | });
151 | dropZone.addEventListener('dragleave', () => {
152 | dropZone.style.borderColor = 'var(--text-secondary)';
153 | });
154 | dropZone.addEventListener('drop', async (e) => {
155 | e.preventDefault();
156 | dropZone.style.borderColor = 'var(--text-secondary)';
157 | await handleInputItems(e.dataTransfer.files);
158 | });
159 |
160 | selectFolderBtn.addEventListener('click', async () => {
161 | const folderPath = await ipcRenderer.invoke('select-input-directory');
162 | if (folderPath) {
163 | const filesFromFolder = getFilesFromDirectory(folderPath);
164 | if (filesFromFolder.length === 0) {
165 | showStatus('No supported files found in the selected folder.', 'error');
166 | } else {
167 | filesFromFolder.forEach((f) => addFile(f));
168 | updateFileListUI();
169 | updateProcessButton();
170 | }
171 | }
172 | });
173 |
174 | clearFilesBtn.addEventListener('click', clearState);
175 | selectOutputBtn.addEventListener('click', async () => {
176 | outputDirectory = await ipcRenderer.invoke('select-output-directory');
177 | if (outputDirectory) {
178 | outputDirInput.value = outputDirectory;
179 | localStorage.setItem('outputDirectory', outputDirectory);
180 | outputLinkDiv.classList.remove('hidden');
181 | openOutputFolderLink.onclick = async () => {
182 | await ipcRenderer.invoke('open-folder', outputDirectory);
183 | };
184 | }
185 | updateProcessButton();
186 | });
187 |
188 | processButton.addEventListener('click', processFiles);
189 |
190 | async function processFiles() {
191 | if (selectedFiles.length === 0) return;
192 |
193 | if (!userState.isPro) {
194 | checkDailyReset();
195 | if (userState.dailyCount >= 100) {
196 | showStatus(`You've reached your 100-file daily limit. Upgrade to Pro or wait until midnight for a limit reset.`, 'error');
197 | return;
198 | }
199 | }
200 |
201 | processButton.disabled = true;
202 | const oldButtonText = processButton.innerHTML;
203 | processButton.innerHTML = ` Anonymizing...`;
204 |
205 | progress.classList.remove('hidden');
206 | let total = selectedFiles.length;
207 | let processedCount = 0;
208 |
209 | for (let i = 0; i < total; i++) {
210 | if (!userState.isPro) {
211 | userState.dailyCount++;
212 | saveUserState();
213 | if (userState.dailyCount > 100) {
214 | showStatus(`You've reached your 100-file daily limit mid-batch.`, 'error');
215 | break;
216 | }
217 | }
218 | const file = selectedFiles[i];
219 | const result = await ipcRenderer.invoke('process-file', {
220 | filePath: file.path,
221 | outputDir: outputDirectory
222 | });
223 | if (!result.success) {
224 | showStatus(`Error processing ${file.name}: ${result.error}`, 'error');
225 | }
226 | processedCount++;
227 | let percentage = Math.floor((processedCount / total) * 100);
228 | progressBar.style.width = `${percentage}%`;
229 | }
230 |
231 | progressBar.style.width = '100%';
232 | showStatus(`Files processed successfully!`, 'success');
233 |
234 | processButton.disabled = false;
235 | processButton.innerHTML = oldButtonText;
236 |
237 | if (outputDirectory) {
238 | outputLinkDiv.classList.remove('hidden');
239 | openOutputFolderLink.onclick = async () => {
240 | await ipcRenderer.invoke('open-folder', outputDirectory);
241 | };
242 | }
243 |
244 | window.scrollTo(0, document.body.scrollHeight);
245 | clearState();
246 |
247 | setTimeout(() => {
248 | progress.classList.add('hidden');
249 | progressBar.style.width = '0%';
250 | }, 1500);
251 | }
252 |
253 | // Basic file logic
254 | async function handleInputItems(fileList) {
255 | for (let i = 0; i < fileList.length; i++) {
256 | let fileItem = fileList[i];
257 | if (!fileItem.path) {
258 | fileItem = await createTempFile(fileItem);
259 | if (!fileItem) continue;
260 | }
261 | try {
262 | const stats = fs.lstatSync(fileItem.path);
263 | if (stats.isDirectory()) {
264 | const filesFromFolder = getFilesFromDirectory(fileItem.path);
265 | filesFromFolder.forEach((f) => addFile(f));
266 | } else {
267 | addFile({ path: fileItem.path, name: fileItem.name });
268 | }
269 | } catch (error) {
270 | console.error(error);
271 | showStatus(`Error processing ${fileItem.name}: ${error.message}`, 'error');
272 | }
273 | }
274 | updateFileListUI();
275 | updateProcessButton();
276 | }
277 |
278 | function createTempFile(fileItem) {
279 | return new Promise((resolve) => {
280 | const reader = new FileReader();
281 | reader.onload = () => {
282 | const buffer = Buffer.from(reader.result);
283 | const tempDir = os.tmpdir();
284 | const tempPath = path.join(tempDir, fileItem.name);
285 | fs.writeFile(tempPath, buffer, (err) => {
286 | if (err) {
287 | showStatus(`Error writing temporary file: ${err.message}`, 'error');
288 | resolve(null);
289 | } else {
290 | resolve({ path: tempPath, name: fileItem.name });
291 | }
292 | });
293 | };
294 | reader.onerror = () => {
295 | showStatus(`Error reading file ${fileItem.name}`, 'error');
296 | resolve(null);
297 | };
298 | reader.readAsArrayBuffer(fileItem);
299 | });
300 | }
301 |
302 | function getFilesFromDirectory(dirPath) {
303 | let results = [];
304 | try {
305 | const list = fs.readdirSync(dirPath);
306 | list.forEach((file) => {
307 | const filePath = path.join(dirPath, file);
308 | const stat = fs.statSync(filePath);
309 | if (stat && stat.isDirectory()) {
310 | results = results.concat(getFilesFromDirectory(filePath));
311 | } else {
312 | const ext = path.extname(file).toLowerCase();
313 | if (['.doc','.docx','.xls','.xlsx','.csv','.pdf','.txt'].includes(ext)) {
314 | results.push({ path: filePath, name: file });
315 | }
316 | }
317 | });
318 | } catch (err) {
319 | console.error(`Error reading directory ${dirPath}: ${err.message}`);
320 | }
321 | return results;
322 | }
323 |
324 | function addFile(fileObj) {
325 | const ext = path.extname(fileObj.path).toLowerCase();
326 | if (['.doc','.docx','.xls','.xlsx','.csv','.pdf','.txt'].includes(ext)) {
327 | if (!selectedFiles.find((f) => f.path === fileObj.path)) {
328 | selectedFiles.push(fileObj);
329 | }
330 | } else {
331 | showStatus(`Unsupported file type: ${fileObj.name}`, 'error');
332 | }
333 | }
334 |
335 | function updateFileListUI() {
336 | filesUl.innerHTML = '';
337 | if (selectedFiles.length === 0) {
338 | fileListDiv.classList.add('hidden');
339 | return;
340 | }
341 | fileListDiv.classList.remove('hidden');
342 | selectedFiles.forEach(file => {
343 | const li = document.createElement('li');
344 | li.textContent = file.name;
345 | filesUl.appendChild(li);
346 | });
347 | }
348 |
349 | function updateProcessButton() {
350 | processButton.disabled = (selectedFiles.length === 0);
351 | }
352 |
353 | function clearState() {
354 | selectedFiles = [];
355 | fileInput.value = '';
356 | updateFileListUI();
357 | updateProcessButton();
358 | }
359 |
360 | // Show status in main area
361 | function showStatus(message, type) {
362 | statusDiv.textContent = message;
363 | statusDiv.className = `status ${type}`;
364 | statusDiv.classList.remove('hidden');
365 | }
366 |
367 | // Logs from main -> renderer
368 | ipcRenderer.on('log-message', (event, msg) => {
369 | logArea.classList.remove('hidden');
370 | logMessages.textContent = `Status: ${msg}`;
371 | });
372 |
373 |
374 |
375 | // PRO Upgrade UI & Logic
376 | proButton.addEventListener('click', () => {
377 | if (userState.isPro) {
378 | showManageModal();
379 | } else {
380 | showUpgradeModal();
381 | }
382 | });
383 |
384 | proInfoLink.addEventListener('click', () => {
385 | if (userState.isPro) {
386 | showManageModal();
387 | } else {
388 | showInfoModal();
389 | }
390 | });
391 |
392 | // Upgrade modal
393 | upgradeClose.addEventListener('click', hideUpgradeModal);
394 | copyDeviceIdBtn.addEventListener('click', () => {
395 | deviceIdField.select();
396 | document.execCommand('copy');
397 | showStatus('Device ID copied to clipboard!', 'success');
398 | });
399 |
400 | // White wide button => open external store
401 | upgradeStoreBtn.addEventListener('click', () => {
402 | require('electron').shell.openExternal('https://amicus5.com/store/PA');
403 | });
404 |
405 | // Validate key => show message in #key-message
406 | validateKeyBtn.addEventListener('click', () => {
407 | const key = proKeyInput.value.trim();
408 | if (!key) {
409 | showKeyMessage('Please enter a key.', 'error');
410 | return;
411 | }
412 | if (validateProKey(key)) {
413 | userState.isPro = true;
414 | saveUserState();
415 | hideUpgradeModal();
416 | showStatus('Pro activated! Enjoy unlimited usage.', 'success');
417 | updateProUI();
418 | } else {
419 | showKeyMessage('Invalid key.', 'error');
420 | }
421 | });
422 |
423 | // Manage Plan modal
424 | manageClose.addEventListener('click', hideManageModal);
425 | downgradeBtn.addEventListener('click', () => {
426 | userState.isPro = false;
427 | saveUserState();
428 | hideManageModal();
429 | showStatus('You have been downgraded to free plan.', 'success');
430 | updateProUI();
431 | });
432 |
433 | // Info modal
434 | infoClose.addEventListener('click', hideInfoModal);
435 |
436 | // Simple validation
437 | function validateProKey(key) {
438 | // e.g. "MASTERTESTKEY" or check simpleHash(userState.deviceID)
439 | // Return true if matches
440 | return (key === 'MASTERTESTKEY');
441 | }
442 |
443 | function showUpgradeModal() {
444 | deviceIdField.value = userState.deviceID;
445 | keyMessageDiv.classList.add('hidden'); // hide old messages
446 | upgradeModal.classList.add('show');
447 | }
448 | function hideUpgradeModal() {
449 | upgradeModal.classList.remove('show');
450 | }
451 | function showInfoModal() {
452 | infoModal.classList.add('show');
453 | }
454 | function hideInfoModal() {
455 | infoModal.classList.remove('show');
456 | }
457 | function showManageModal() {
458 | manageModal.classList.add('show');
459 | }
460 | function hideManageModal() {
461 | manageModal.classList.remove('show');
462 | }
463 |
464 | function showKeyMessage(msg, type) {
465 | keyMessageDiv.textContent = msg;
466 | keyMessageDiv.className = 'key-message'; // reset
467 | keyMessageDiv.classList.add(type === 'error' ? 'error' : 'success');
468 | keyMessageDiv.classList.remove('hidden');
469 | }
470 |
471 | function updateProUI() {
472 | if (userState.isPro) {
473 | proButton.textContent = 'Pro Version';
474 | proButton.classList.remove('pro-upgrade');
475 | proButton.classList.add('pro-active');
476 | proInfoLink.textContent = 'Manage Plan';
477 | } else {
478 | proButton.innerHTML = ` Upgrade to Pro`;
479 | proButton.classList.remove('pro-active');
480 | proButton.classList.add('pro-upgrade');
481 | proInfoLink.textContent = "What's Included in Pro?";
482 | }
483 | }
484 |
--------------------------------------------------------------------------------
/all.min.css:
--------------------------------------------------------------------------------
1 | /*!
2 | * Font Awesome Free 5.15.4 by @fontawesome - https://fontawesome.com
3 | * License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL 1.1, Code: MIT License)
4 | */
5 | .fa,.fab,.fad,.fal,.far,.fas{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;display:inline-block;font-style:normal;font-variant:normal;text-rendering:auto;line-height:1}.fa-lg{font-size:1.33333em;line-height:.75em;vertical-align:-.0667em}.fa-xs{font-size:.75em}.fa-sm{font-size:.875em}.fa-1x{font-size:1em}.fa-2x{font-size:2em}.fa-3x{font-size:3em}.fa-4x{font-size:4em}.fa-5x{font-size:5em}.fa-6x{font-size:6em}.fa-7x{font-size:7em}.fa-8x{font-size:8em}.fa-9x{font-size:9em}.fa-10x{font-size:10em}.fa-fw{text-align:center;width:1.25em}.fa-ul{list-style-type:none;margin-left:2.5em;padding-left:0}.fa-ul>li{position:relative}.fa-li{left:-2em;position:absolute;text-align:center;width:2em;line-height:inherit}.fa-border{border:.08em solid #eee;border-radius:.1em;padding:.2em .25em .15em}.fa-pull-left{float:left}.fa-pull-right{float:right}.fa.fa-pull-left,.fab.fa-pull-left,.fal.fa-pull-left,.far.fa-pull-left,.fas.fa-pull-left{margin-right:.3em}.fa.fa-pull-right,.fab.fa-pull-right,.fal.fa-pull-right,.far.fa-pull-right,.fas.fa-pull-right{margin-left:.3em}.fa-spin{-webkit-animation:fa-spin 2s linear infinite;animation:fa-spin 2s linear infinite}.fa-pulse{-webkit-animation:fa-spin 1s steps(8) infinite;animation:fa-spin 1s steps(8) infinite}@-webkit-keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}}@keyframes fa-spin{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}}.fa-rotate-90{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=1)";-webkit-transform:rotate(90deg);transform:rotate(90deg)}.fa-rotate-180{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2)";-webkit-transform:rotate(180deg);transform:rotate(180deg)}.fa-rotate-270{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=3)";-webkit-transform:rotate(270deg);transform:rotate(270deg)}.fa-flip-horizontal{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=0, mirror=1)";-webkit-transform:scaleX(-1);transform:scaleX(-1)}.fa-flip-vertical{-webkit-transform:scaleY(-1);transform:scaleY(-1)}.fa-flip-both,.fa-flip-horizontal.fa-flip-vertical,.fa-flip-vertical{-ms-filter:"progid:DXImageTransform.Microsoft.BasicImage(rotation=2, mirror=1)"}.fa-flip-both,.fa-flip-horizontal.fa-flip-vertical{-webkit-transform:scale(-1);transform:scale(-1)}:root .fa-flip-both,:root .fa-flip-horizontal,:root .fa-flip-vertical,:root .fa-rotate-90,:root .fa-rotate-180,:root .fa-rotate-270{-webkit-filter:none;filter:none}.fa-stack{display:inline-block;height:2em;line-height:2em;position:relative;vertical-align:middle;width:2.5em}.fa-stack-1x,.fa-stack-2x{left:0;position:absolute;text-align:center;width:100%}.fa-stack-1x{line-height:inherit}.fa-stack-2x{font-size:2em}.fa-inverse{color:#fff}.fa-500px:before{content:"\f26e"}.fa-accessible-icon:before{content:"\f368"}.fa-accusoft:before{content:"\f369"}.fa-acquisitions-incorporated:before{content:"\f6af"}.fa-ad:before{content:"\f641"}.fa-address-book:before{content:"\f2b9"}.fa-address-card:before{content:"\f2bb"}.fa-adjust:before{content:"\f042"}.fa-adn:before{content:"\f170"}.fa-adversal:before{content:"\f36a"}.fa-affiliatetheme:before{content:"\f36b"}.fa-air-freshener:before{content:"\f5d0"}.fa-airbnb:before{content:"\f834"}.fa-algolia:before{content:"\f36c"}.fa-align-center:before{content:"\f037"}.fa-align-justify:before{content:"\f039"}.fa-align-left:before{content:"\f036"}.fa-align-right:before{content:"\f038"}.fa-alipay:before{content:"\f642"}.fa-allergies:before{content:"\f461"}.fa-amazon:before{content:"\f270"}.fa-amazon-pay:before{content:"\f42c"}.fa-ambulance:before{content:"\f0f9"}.fa-american-sign-language-interpreting:before{content:"\f2a3"}.fa-amilia:before{content:"\f36d"}.fa-anchor:before{content:"\f13d"}.fa-android:before{content:"\f17b"}.fa-angellist:before{content:"\f209"}.fa-angle-double-down:before{content:"\f103"}.fa-angle-double-left:before{content:"\f100"}.fa-angle-double-right:before{content:"\f101"}.fa-angle-double-up:before{content:"\f102"}.fa-angle-down:before{content:"\f107"}.fa-angle-left:before{content:"\f104"}.fa-angle-right:before{content:"\f105"}.fa-angle-up:before{content:"\f106"}.fa-angry:before{content:"\f556"}.fa-angrycreative:before{content:"\f36e"}.fa-angular:before{content:"\f420"}.fa-ankh:before{content:"\f644"}.fa-app-store:before{content:"\f36f"}.fa-app-store-ios:before{content:"\f370"}.fa-apper:before{content:"\f371"}.fa-apple:before{content:"\f179"}.fa-apple-alt:before{content:"\f5d1"}.fa-apple-pay:before{content:"\f415"}.fa-archive:before{content:"\f187"}.fa-archway:before{content:"\f557"}.fa-arrow-alt-circle-down:before{content:"\f358"}.fa-arrow-alt-circle-left:before{content:"\f359"}.fa-arrow-alt-circle-right:before{content:"\f35a"}.fa-arrow-alt-circle-up:before{content:"\f35b"}.fa-arrow-circle-down:before{content:"\f0ab"}.fa-arrow-circle-left:before{content:"\f0a8"}.fa-arrow-circle-right:before{content:"\f0a9"}.fa-arrow-circle-up:before{content:"\f0aa"}.fa-arrow-down:before{content:"\f063"}.fa-arrow-left:before{content:"\f060"}.fa-arrow-right:before{content:"\f061"}.fa-arrow-up:before{content:"\f062"}.fa-arrows-alt:before{content:"\f0b2"}.fa-arrows-alt-h:before{content:"\f337"}.fa-arrows-alt-v:before{content:"\f338"}.fa-artstation:before{content:"\f77a"}.fa-assistive-listening-systems:before{content:"\f2a2"}.fa-asterisk:before{content:"\f069"}.fa-asymmetrik:before{content:"\f372"}.fa-at:before{content:"\f1fa"}.fa-atlas:before{content:"\f558"}.fa-atlassian:before{content:"\f77b"}.fa-atom:before{content:"\f5d2"}.fa-audible:before{content:"\f373"}.fa-audio-description:before{content:"\f29e"}.fa-autoprefixer:before{content:"\f41c"}.fa-avianex:before{content:"\f374"}.fa-aviato:before{content:"\f421"}.fa-award:before{content:"\f559"}.fa-aws:before{content:"\f375"}.fa-baby:before{content:"\f77c"}.fa-baby-carriage:before{content:"\f77d"}.fa-backspace:before{content:"\f55a"}.fa-backward:before{content:"\f04a"}.fa-bacon:before{content:"\f7e5"}.fa-bacteria:before{content:"\e059"}.fa-bacterium:before{content:"\e05a"}.fa-bahai:before{content:"\f666"}.fa-balance-scale:before{content:"\f24e"}.fa-balance-scale-left:before{content:"\f515"}.fa-balance-scale-right:before{content:"\f516"}.fa-ban:before{content:"\f05e"}.fa-band-aid:before{content:"\f462"}.fa-bandcamp:before{content:"\f2d5"}.fa-barcode:before{content:"\f02a"}.fa-bars:before{content:"\f0c9"}.fa-baseball-ball:before{content:"\f433"}.fa-basketball-ball:before{content:"\f434"}.fa-bath:before{content:"\f2cd"}.fa-battery-empty:before{content:"\f244"}.fa-battery-full:before{content:"\f240"}.fa-battery-half:before{content:"\f242"}.fa-battery-quarter:before{content:"\f243"}.fa-battery-three-quarters:before{content:"\f241"}.fa-battle-net:before{content:"\f835"}.fa-bed:before{content:"\f236"}.fa-beer:before{content:"\f0fc"}.fa-behance:before{content:"\f1b4"}.fa-behance-square:before{content:"\f1b5"}.fa-bell:before{content:"\f0f3"}.fa-bell-slash:before{content:"\f1f6"}.fa-bezier-curve:before{content:"\f55b"}.fa-bible:before{content:"\f647"}.fa-bicycle:before{content:"\f206"}.fa-biking:before{content:"\f84a"}.fa-bimobject:before{content:"\f378"}.fa-binoculars:before{content:"\f1e5"}.fa-biohazard:before{content:"\f780"}.fa-birthday-cake:before{content:"\f1fd"}.fa-bitbucket:before{content:"\f171"}.fa-bitcoin:before{content:"\f379"}.fa-bity:before{content:"\f37a"}.fa-black-tie:before{content:"\f27e"}.fa-blackberry:before{content:"\f37b"}.fa-blender:before{content:"\f517"}.fa-blender-phone:before{content:"\f6b6"}.fa-blind:before{content:"\f29d"}.fa-blog:before{content:"\f781"}.fa-blogger:before{content:"\f37c"}.fa-blogger-b:before{content:"\f37d"}.fa-bluetooth:before{content:"\f293"}.fa-bluetooth-b:before{content:"\f294"}.fa-bold:before{content:"\f032"}.fa-bolt:before{content:"\f0e7"}.fa-bomb:before{content:"\f1e2"}.fa-bone:before{content:"\f5d7"}.fa-bong:before{content:"\f55c"}.fa-book:before{content:"\f02d"}.fa-book-dead:before{content:"\f6b7"}.fa-book-medical:before{content:"\f7e6"}.fa-book-open:before{content:"\f518"}.fa-book-reader:before{content:"\f5da"}.fa-bookmark:before{content:"\f02e"}.fa-bootstrap:before{content:"\f836"}.fa-border-all:before{content:"\f84c"}.fa-border-none:before{content:"\f850"}.fa-border-style:before{content:"\f853"}.fa-bowling-ball:before{content:"\f436"}.fa-box:before{content:"\f466"}.fa-box-open:before{content:"\f49e"}.fa-box-tissue:before{content:"\e05b"}.fa-boxes:before{content:"\f468"}.fa-braille:before{content:"\f2a1"}.fa-brain:before{content:"\f5dc"}.fa-bread-slice:before{content:"\f7ec"}.fa-briefcase:before{content:"\f0b1"}.fa-briefcase-medical:before{content:"\f469"}.fa-broadcast-tower:before{content:"\f519"}.fa-broom:before{content:"\f51a"}.fa-brush:before{content:"\f55d"}.fa-btc:before{content:"\f15a"}.fa-buffer:before{content:"\f837"}.fa-bug:before{content:"\f188"}.fa-building:before{content:"\f1ad"}.fa-bullhorn:before{content:"\f0a1"}.fa-bullseye:before{content:"\f140"}.fa-burn:before{content:"\f46a"}.fa-buromobelexperte:before{content:"\f37f"}.fa-bus:before{content:"\f207"}.fa-bus-alt:before{content:"\f55e"}.fa-business-time:before{content:"\f64a"}.fa-buy-n-large:before{content:"\f8a6"}.fa-buysellads:before{content:"\f20d"}.fa-calculator:before{content:"\f1ec"}.fa-calendar:before{content:"\f133"}.fa-calendar-alt:before{content:"\f073"}.fa-calendar-check:before{content:"\f274"}.fa-calendar-day:before{content:"\f783"}.fa-calendar-minus:before{content:"\f272"}.fa-calendar-plus:before{content:"\f271"}.fa-calendar-times:before{content:"\f273"}.fa-calendar-week:before{content:"\f784"}.fa-camera:before{content:"\f030"}.fa-camera-retro:before{content:"\f083"}.fa-campground:before{content:"\f6bb"}.fa-canadian-maple-leaf:before{content:"\f785"}.fa-candy-cane:before{content:"\f786"}.fa-cannabis:before{content:"\f55f"}.fa-capsules:before{content:"\f46b"}.fa-car:before{content:"\f1b9"}.fa-car-alt:before{content:"\f5de"}.fa-car-battery:before{content:"\f5df"}.fa-car-crash:before{content:"\f5e1"}.fa-car-side:before{content:"\f5e4"}.fa-caravan:before{content:"\f8ff"}.fa-caret-down:before{content:"\f0d7"}.fa-caret-left:before{content:"\f0d9"}.fa-caret-right:before{content:"\f0da"}.fa-caret-square-down:before{content:"\f150"}.fa-caret-square-left:before{content:"\f191"}.fa-caret-square-right:before{content:"\f152"}.fa-caret-square-up:before{content:"\f151"}.fa-caret-up:before{content:"\f0d8"}.fa-carrot:before{content:"\f787"}.fa-cart-arrow-down:before{content:"\f218"}.fa-cart-plus:before{content:"\f217"}.fa-cash-register:before{content:"\f788"}.fa-cat:before{content:"\f6be"}.fa-cc-amazon-pay:before{content:"\f42d"}.fa-cc-amex:before{content:"\f1f3"}.fa-cc-apple-pay:before{content:"\f416"}.fa-cc-diners-club:before{content:"\f24c"}.fa-cc-discover:before{content:"\f1f2"}.fa-cc-jcb:before{content:"\f24b"}.fa-cc-mastercard:before{content:"\f1f1"}.fa-cc-paypal:before{content:"\f1f4"}.fa-cc-stripe:before{content:"\f1f5"}.fa-cc-visa:before{content:"\f1f0"}.fa-centercode:before{content:"\f380"}.fa-centos:before{content:"\f789"}.fa-certificate:before{content:"\f0a3"}.fa-chair:before{content:"\f6c0"}.fa-chalkboard:before{content:"\f51b"}.fa-chalkboard-teacher:before{content:"\f51c"}.fa-charging-station:before{content:"\f5e7"}.fa-chart-area:before{content:"\f1fe"}.fa-chart-bar:before{content:"\f080"}.fa-chart-line:before{content:"\f201"}.fa-chart-pie:before{content:"\f200"}.fa-check:before{content:"\f00c"}.fa-check-circle:before{content:"\f058"}.fa-check-double:before{content:"\f560"}.fa-check-square:before{content:"\f14a"}.fa-cheese:before{content:"\f7ef"}.fa-chess:before{content:"\f439"}.fa-chess-bishop:before{content:"\f43a"}.fa-chess-board:before{content:"\f43c"}.fa-chess-king:before{content:"\f43f"}.fa-chess-knight:before{content:"\f441"}.fa-chess-pawn:before{content:"\f443"}.fa-chess-queen:before{content:"\f445"}.fa-chess-rook:before{content:"\f447"}.fa-chevron-circle-down:before{content:"\f13a"}.fa-chevron-circle-left:before{content:"\f137"}.fa-chevron-circle-right:before{content:"\f138"}.fa-chevron-circle-up:before{content:"\f139"}.fa-chevron-down:before{content:"\f078"}.fa-chevron-left:before{content:"\f053"}.fa-chevron-right:before{content:"\f054"}.fa-chevron-up:before{content:"\f077"}.fa-child:before{content:"\f1ae"}.fa-chrome:before{content:"\f268"}.fa-chromecast:before{content:"\f838"}.fa-church:before{content:"\f51d"}.fa-circle:before{content:"\f111"}.fa-circle-notch:before{content:"\f1ce"}.fa-city:before{content:"\f64f"}.fa-clinic-medical:before{content:"\f7f2"}.fa-clipboard:before{content:"\f328"}.fa-clipboard-check:before{content:"\f46c"}.fa-clipboard-list:before{content:"\f46d"}.fa-clock:before{content:"\f017"}.fa-clone:before{content:"\f24d"}.fa-closed-captioning:before{content:"\f20a"}.fa-cloud:before{content:"\f0c2"}.fa-cloud-download-alt:before{content:"\f381"}.fa-cloud-meatball:before{content:"\f73b"}.fa-cloud-moon:before{content:"\f6c3"}.fa-cloud-moon-rain:before{content:"\f73c"}.fa-cloud-rain:before{content:"\f73d"}.fa-cloud-showers-heavy:before{content:"\f740"}.fa-cloud-sun:before{content:"\f6c4"}.fa-cloud-sun-rain:before{content:"\f743"}.fa-cloud-upload-alt:before{content:"\f382"}.fa-cloudflare:before{content:"\e07d"}.fa-cloudscale:before{content:"\f383"}.fa-cloudsmith:before{content:"\f384"}.fa-cloudversify:before{content:"\f385"}.fa-cocktail:before{content:"\f561"}.fa-code:before{content:"\f121"}.fa-code-branch:before{content:"\f126"}.fa-codepen:before{content:"\f1cb"}.fa-codiepie:before{content:"\f284"}.fa-coffee:before{content:"\f0f4"}.fa-cog:before{content:"\f013"}.fa-cogs:before{content:"\f085"}.fa-coins:before{content:"\f51e"}.fa-columns:before{content:"\f0db"}.fa-comment:before{content:"\f075"}.fa-comment-alt:before{content:"\f27a"}.fa-comment-dollar:before{content:"\f651"}.fa-comment-dots:before{content:"\f4ad"}.fa-comment-medical:before{content:"\f7f5"}.fa-comment-slash:before{content:"\f4b3"}.fa-comments:before{content:"\f086"}.fa-comments-dollar:before{content:"\f653"}.fa-compact-disc:before{content:"\f51f"}.fa-compass:before{content:"\f14e"}.fa-compress:before{content:"\f066"}.fa-compress-alt:before{content:"\f422"}.fa-compress-arrows-alt:before{content:"\f78c"}.fa-concierge-bell:before{content:"\f562"}.fa-confluence:before{content:"\f78d"}.fa-connectdevelop:before{content:"\f20e"}.fa-contao:before{content:"\f26d"}.fa-cookie:before{content:"\f563"}.fa-cookie-bite:before{content:"\f564"}.fa-copy:before{content:"\f0c5"}.fa-copyright:before{content:"\f1f9"}.fa-cotton-bureau:before{content:"\f89e"}.fa-couch:before{content:"\f4b8"}.fa-cpanel:before{content:"\f388"}.fa-creative-commons:before{content:"\f25e"}.fa-creative-commons-by:before{content:"\f4e7"}.fa-creative-commons-nc:before{content:"\f4e8"}.fa-creative-commons-nc-eu:before{content:"\f4e9"}.fa-creative-commons-nc-jp:before{content:"\f4ea"}.fa-creative-commons-nd:before{content:"\f4eb"}.fa-creative-commons-pd:before{content:"\f4ec"}.fa-creative-commons-pd-alt:before{content:"\f4ed"}.fa-creative-commons-remix:before{content:"\f4ee"}.fa-creative-commons-sa:before{content:"\f4ef"}.fa-creative-commons-sampling:before{content:"\f4f0"}.fa-creative-commons-sampling-plus:before{content:"\f4f1"}.fa-creative-commons-share:before{content:"\f4f2"}.fa-creative-commons-zero:before{content:"\f4f3"}.fa-credit-card:before{content:"\f09d"}.fa-critical-role:before{content:"\f6c9"}.fa-crop:before{content:"\f125"}.fa-crop-alt:before{content:"\f565"}.fa-cross:before{content:"\f654"}.fa-crosshairs:before{content:"\f05b"}.fa-crow:before{content:"\f520"}.fa-crown:before{content:"\f521"}.fa-crutch:before{content:"\f7f7"}.fa-css3:before{content:"\f13c"}.fa-css3-alt:before{content:"\f38b"}.fa-cube:before{content:"\f1b2"}.fa-cubes:before{content:"\f1b3"}.fa-cut:before{content:"\f0c4"}.fa-cuttlefish:before{content:"\f38c"}.fa-d-and-d:before{content:"\f38d"}.fa-d-and-d-beyond:before{content:"\f6ca"}.fa-dailymotion:before{content:"\e052"}.fa-dashcube:before{content:"\f210"}.fa-database:before{content:"\f1c0"}.fa-deaf:before{content:"\f2a4"}.fa-deezer:before{content:"\e077"}.fa-delicious:before{content:"\f1a5"}.fa-democrat:before{content:"\f747"}.fa-deploydog:before{content:"\f38e"}.fa-deskpro:before{content:"\f38f"}.fa-desktop:before{content:"\f108"}.fa-dev:before{content:"\f6cc"}.fa-deviantart:before{content:"\f1bd"}.fa-dharmachakra:before{content:"\f655"}.fa-dhl:before{content:"\f790"}.fa-diagnoses:before{content:"\f470"}.fa-diaspora:before{content:"\f791"}.fa-dice:before{content:"\f522"}.fa-dice-d20:before{content:"\f6cf"}.fa-dice-d6:before{content:"\f6d1"}.fa-dice-five:before{content:"\f523"}.fa-dice-four:before{content:"\f524"}.fa-dice-one:before{content:"\f525"}.fa-dice-six:before{content:"\f526"}.fa-dice-three:before{content:"\f527"}.fa-dice-two:before{content:"\f528"}.fa-digg:before{content:"\f1a6"}.fa-digital-ocean:before{content:"\f391"}.fa-digital-tachograph:before{content:"\f566"}.fa-directions:before{content:"\f5eb"}.fa-discord:before{content:"\f392"}.fa-discourse:before{content:"\f393"}.fa-disease:before{content:"\f7fa"}.fa-divide:before{content:"\f529"}.fa-dizzy:before{content:"\f567"}.fa-dna:before{content:"\f471"}.fa-dochub:before{content:"\f394"}.fa-docker:before{content:"\f395"}.fa-dog:before{content:"\f6d3"}.fa-dollar-sign:before{content:"\f155"}.fa-dolly:before{content:"\f472"}.fa-dolly-flatbed:before{content:"\f474"}.fa-donate:before{content:"\f4b9"}.fa-door-closed:before{content:"\f52a"}.fa-door-open:before{content:"\f52b"}.fa-dot-circle:before{content:"\f192"}.fa-dove:before{content:"\f4ba"}.fa-download:before{content:"\f019"}.fa-draft2digital:before{content:"\f396"}.fa-drafting-compass:before{content:"\f568"}.fa-dragon:before{content:"\f6d5"}.fa-draw-polygon:before{content:"\f5ee"}.fa-dribbble:before{content:"\f17d"}.fa-dribbble-square:before{content:"\f397"}.fa-dropbox:before{content:"\f16b"}.fa-drum:before{content:"\f569"}.fa-drum-steelpan:before{content:"\f56a"}.fa-drumstick-bite:before{content:"\f6d7"}.fa-drupal:before{content:"\f1a9"}.fa-dumbbell:before{content:"\f44b"}.fa-dumpster:before{content:"\f793"}.fa-dumpster-fire:before{content:"\f794"}.fa-dungeon:before{content:"\f6d9"}.fa-dyalog:before{content:"\f399"}.fa-earlybirds:before{content:"\f39a"}.fa-ebay:before{content:"\f4f4"}.fa-edge:before{content:"\f282"}.fa-edge-legacy:before{content:"\e078"}.fa-edit:before{content:"\f044"}.fa-egg:before{content:"\f7fb"}.fa-eject:before{content:"\f052"}.fa-elementor:before{content:"\f430"}.fa-ellipsis-h:before{content:"\f141"}.fa-ellipsis-v:before{content:"\f142"}.fa-ello:before{content:"\f5f1"}.fa-ember:before{content:"\f423"}.fa-empire:before{content:"\f1d1"}.fa-envelope:before{content:"\f0e0"}.fa-envelope-open:before{content:"\f2b6"}.fa-envelope-open-text:before{content:"\f658"}.fa-envelope-square:before{content:"\f199"}.fa-envira:before{content:"\f299"}.fa-equals:before{content:"\f52c"}.fa-eraser:before{content:"\f12d"}.fa-erlang:before{content:"\f39d"}.fa-ethereum:before{content:"\f42e"}.fa-ethernet:before{content:"\f796"}.fa-etsy:before{content:"\f2d7"}.fa-euro-sign:before{content:"\f153"}.fa-evernote:before{content:"\f839"}.fa-exchange-alt:before{content:"\f362"}.fa-exclamation:before{content:"\f12a"}.fa-exclamation-circle:before{content:"\f06a"}.fa-exclamation-triangle:before{content:"\f071"}.fa-expand:before{content:"\f065"}.fa-expand-alt:before{content:"\f424"}.fa-expand-arrows-alt:before{content:"\f31e"}.fa-expeditedssl:before{content:"\f23e"}.fa-external-link-alt:before{content:"\f35d"}.fa-external-link-square-alt:before{content:"\f360"}.fa-eye:before{content:"\f06e"}.fa-eye-dropper:before{content:"\f1fb"}.fa-eye-slash:before{content:"\f070"}.fa-facebook:before{content:"\f09a"}.fa-facebook-f:before{content:"\f39e"}.fa-facebook-messenger:before{content:"\f39f"}.fa-facebook-square:before{content:"\f082"}.fa-fan:before{content:"\f863"}.fa-fantasy-flight-games:before{content:"\f6dc"}.fa-fast-backward:before{content:"\f049"}.fa-fast-forward:before{content:"\f050"}.fa-faucet:before{content:"\e005"}.fa-fax:before{content:"\f1ac"}.fa-feather:before{content:"\f52d"}.fa-feather-alt:before{content:"\f56b"}.fa-fedex:before{content:"\f797"}.fa-fedora:before{content:"\f798"}.fa-female:before{content:"\f182"}.fa-fighter-jet:before{content:"\f0fb"}.fa-figma:before{content:"\f799"}.fa-file:before{content:"\f15b"}.fa-file-alt:before{content:"\f15c"}.fa-file-archive:before{content:"\f1c6"}.fa-file-audio:before{content:"\f1c7"}.fa-file-code:before{content:"\f1c9"}.fa-file-contract:before{content:"\f56c"}.fa-file-csv:before{content:"\f6dd"}.fa-file-download:before{content:"\f56d"}.fa-file-excel:before{content:"\f1c3"}.fa-file-export:before{content:"\f56e"}.fa-file-image:before{content:"\f1c5"}.fa-file-import:before{content:"\f56f"}.fa-file-invoice:before{content:"\f570"}.fa-file-invoice-dollar:before{content:"\f571"}.fa-file-medical:before{content:"\f477"}.fa-file-medical-alt:before{content:"\f478"}.fa-file-pdf:before{content:"\f1c1"}.fa-file-powerpoint:before{content:"\f1c4"}.fa-file-prescription:before{content:"\f572"}.fa-file-signature:before{content:"\f573"}.fa-file-upload:before{content:"\f574"}.fa-file-video:before{content:"\f1c8"}.fa-file-word:before{content:"\f1c2"}.fa-fill:before{content:"\f575"}.fa-fill-drip:before{content:"\f576"}.fa-film:before{content:"\f008"}.fa-filter:before{content:"\f0b0"}.fa-fingerprint:before{content:"\f577"}.fa-fire:before{content:"\f06d"}.fa-fire-alt:before{content:"\f7e4"}.fa-fire-extinguisher:before{content:"\f134"}.fa-firefox:before{content:"\f269"}.fa-firefox-browser:before{content:"\e007"}.fa-first-aid:before{content:"\f479"}.fa-first-order:before{content:"\f2b0"}.fa-first-order-alt:before{content:"\f50a"}.fa-firstdraft:before{content:"\f3a1"}.fa-fish:before{content:"\f578"}.fa-fist-raised:before{content:"\f6de"}.fa-flag:before{content:"\f024"}.fa-flag-checkered:before{content:"\f11e"}.fa-flag-usa:before{content:"\f74d"}.fa-flask:before{content:"\f0c3"}.fa-flickr:before{content:"\f16e"}.fa-flipboard:before{content:"\f44d"}.fa-flushed:before{content:"\f579"}.fa-fly:before{content:"\f417"}.fa-folder:before{content:"\f07b"}.fa-folder-minus:before{content:"\f65d"}.fa-folder-open:before{content:"\f07c"}.fa-folder-plus:before{content:"\f65e"}.fa-font:before{content:"\f031"}.fa-font-awesome:before{content:"\f2b4"}.fa-font-awesome-alt:before{content:"\f35c"}.fa-font-awesome-flag:before{content:"\f425"}.fa-font-awesome-logo-full:before{content:"\f4e6"}.fa-fonticons:before{content:"\f280"}.fa-fonticons-fi:before{content:"\f3a2"}.fa-football-ball:before{content:"\f44e"}.fa-fort-awesome:before{content:"\f286"}.fa-fort-awesome-alt:before{content:"\f3a3"}.fa-forumbee:before{content:"\f211"}.fa-forward:before{content:"\f04e"}.fa-foursquare:before{content:"\f180"}.fa-free-code-camp:before{content:"\f2c5"}.fa-freebsd:before{content:"\f3a4"}.fa-frog:before{content:"\f52e"}.fa-frown:before{content:"\f119"}.fa-frown-open:before{content:"\f57a"}.fa-fulcrum:before{content:"\f50b"}.fa-funnel-dollar:before{content:"\f662"}.fa-futbol:before{content:"\f1e3"}.fa-galactic-republic:before{content:"\f50c"}.fa-galactic-senate:before{content:"\f50d"}.fa-gamepad:before{content:"\f11b"}.fa-gas-pump:before{content:"\f52f"}.fa-gavel:before{content:"\f0e3"}.fa-gem:before{content:"\f3a5"}.fa-genderless:before{content:"\f22d"}.fa-get-pocket:before{content:"\f265"}.fa-gg:before{content:"\f260"}.fa-gg-circle:before{content:"\f261"}.fa-ghost:before{content:"\f6e2"}.fa-gift:before{content:"\f06b"}.fa-gifts:before{content:"\f79c"}.fa-git:before{content:"\f1d3"}.fa-git-alt:before{content:"\f841"}.fa-git-square:before{content:"\f1d2"}.fa-github:before{content:"\f09b"}.fa-github-alt:before{content:"\f113"}.fa-github-square:before{content:"\f092"}.fa-gitkraken:before{content:"\f3a6"}.fa-gitlab:before{content:"\f296"}.fa-gitter:before{content:"\f426"}.fa-glass-cheers:before{content:"\f79f"}.fa-glass-martini:before{content:"\f000"}.fa-glass-martini-alt:before{content:"\f57b"}.fa-glass-whiskey:before{content:"\f7a0"}.fa-glasses:before{content:"\f530"}.fa-glide:before{content:"\f2a5"}.fa-glide-g:before{content:"\f2a6"}.fa-globe:before{content:"\f0ac"}.fa-globe-africa:before{content:"\f57c"}.fa-globe-americas:before{content:"\f57d"}.fa-globe-asia:before{content:"\f57e"}.fa-globe-europe:before{content:"\f7a2"}.fa-gofore:before{content:"\f3a7"}.fa-golf-ball:before{content:"\f450"}.fa-goodreads:before{content:"\f3a8"}.fa-goodreads-g:before{content:"\f3a9"}.fa-google:before{content:"\f1a0"}.fa-google-drive:before{content:"\f3aa"}.fa-google-pay:before{content:"\e079"}.fa-google-play:before{content:"\f3ab"}.fa-google-plus:before{content:"\f2b3"}.fa-google-plus-g:before{content:"\f0d5"}.fa-google-plus-square:before{content:"\f0d4"}.fa-google-wallet:before{content:"\f1ee"}.fa-gopuram:before{content:"\f664"}.fa-graduation-cap:before{content:"\f19d"}.fa-gratipay:before{content:"\f184"}.fa-grav:before{content:"\f2d6"}.fa-greater-than:before{content:"\f531"}.fa-greater-than-equal:before{content:"\f532"}.fa-grimace:before{content:"\f57f"}.fa-grin:before{content:"\f580"}.fa-grin-alt:before{content:"\f581"}.fa-grin-beam:before{content:"\f582"}.fa-grin-beam-sweat:before{content:"\f583"}.fa-grin-hearts:before{content:"\f584"}.fa-grin-squint:before{content:"\f585"}.fa-grin-squint-tears:before{content:"\f586"}.fa-grin-stars:before{content:"\f587"}.fa-grin-tears:before{content:"\f588"}.fa-grin-tongue:before{content:"\f589"}.fa-grin-tongue-squint:before{content:"\f58a"}.fa-grin-tongue-wink:before{content:"\f58b"}.fa-grin-wink:before{content:"\f58c"}.fa-grip-horizontal:before{content:"\f58d"}.fa-grip-lines:before{content:"\f7a4"}.fa-grip-lines-vertical:before{content:"\f7a5"}.fa-grip-vertical:before{content:"\f58e"}.fa-gripfire:before{content:"\f3ac"}.fa-grunt:before{content:"\f3ad"}.fa-guilded:before{content:"\e07e"}.fa-guitar:before{content:"\f7a6"}.fa-gulp:before{content:"\f3ae"}.fa-h-square:before{content:"\f0fd"}.fa-hacker-news:before{content:"\f1d4"}.fa-hacker-news-square:before{content:"\f3af"}.fa-hackerrank:before{content:"\f5f7"}.fa-hamburger:before{content:"\f805"}.fa-hammer:before{content:"\f6e3"}.fa-hamsa:before{content:"\f665"}.fa-hand-holding:before{content:"\f4bd"}.fa-hand-holding-heart:before{content:"\f4be"}.fa-hand-holding-medical:before{content:"\e05c"}.fa-hand-holding-usd:before{content:"\f4c0"}.fa-hand-holding-water:before{content:"\f4c1"}.fa-hand-lizard:before{content:"\f258"}.fa-hand-middle-finger:before{content:"\f806"}.fa-hand-paper:before{content:"\f256"}.fa-hand-peace:before{content:"\f25b"}.fa-hand-point-down:before{content:"\f0a7"}.fa-hand-point-left:before{content:"\f0a5"}.fa-hand-point-right:before{content:"\f0a4"}.fa-hand-point-up:before{content:"\f0a6"}.fa-hand-pointer:before{content:"\f25a"}.fa-hand-rock:before{content:"\f255"}.fa-hand-scissors:before{content:"\f257"}.fa-hand-sparkles:before{content:"\e05d"}.fa-hand-spock:before{content:"\f259"}.fa-hands:before{content:"\f4c2"}.fa-hands-helping:before{content:"\f4c4"}.fa-hands-wash:before{content:"\e05e"}.fa-handshake:before{content:"\f2b5"}.fa-handshake-alt-slash:before{content:"\e05f"}.fa-handshake-slash:before{content:"\e060"}.fa-hanukiah:before{content:"\f6e6"}.fa-hard-hat:before{content:"\f807"}.fa-hashtag:before{content:"\f292"}.fa-hat-cowboy:before{content:"\f8c0"}.fa-hat-cowboy-side:before{content:"\f8c1"}.fa-hat-wizard:before{content:"\f6e8"}.fa-hdd:before{content:"\f0a0"}.fa-head-side-cough:before{content:"\e061"}.fa-head-side-cough-slash:before{content:"\e062"}.fa-head-side-mask:before{content:"\e063"}.fa-head-side-virus:before{content:"\e064"}.fa-heading:before{content:"\f1dc"}.fa-headphones:before{content:"\f025"}.fa-headphones-alt:before{content:"\f58f"}.fa-headset:before{content:"\f590"}.fa-heart:before{content:"\f004"}.fa-heart-broken:before{content:"\f7a9"}.fa-heartbeat:before{content:"\f21e"}.fa-helicopter:before{content:"\f533"}.fa-highlighter:before{content:"\f591"}.fa-hiking:before{content:"\f6ec"}.fa-hippo:before{content:"\f6ed"}.fa-hips:before{content:"\f452"}.fa-hire-a-helper:before{content:"\f3b0"}.fa-history:before{content:"\f1da"}.fa-hive:before{content:"\e07f"}.fa-hockey-puck:before{content:"\f453"}.fa-holly-berry:before{content:"\f7aa"}.fa-home:before{content:"\f015"}.fa-hooli:before{content:"\f427"}.fa-hornbill:before{content:"\f592"}.fa-horse:before{content:"\f6f0"}.fa-horse-head:before{content:"\f7ab"}.fa-hospital:before{content:"\f0f8"}.fa-hospital-alt:before{content:"\f47d"}.fa-hospital-symbol:before{content:"\f47e"}.fa-hospital-user:before{content:"\f80d"}.fa-hot-tub:before{content:"\f593"}.fa-hotdog:before{content:"\f80f"}.fa-hotel:before{content:"\f594"}.fa-hotjar:before{content:"\f3b1"}.fa-hourglass:before{content:"\f254"}.fa-hourglass-end:before{content:"\f253"}.fa-hourglass-half:before{content:"\f252"}.fa-hourglass-start:before{content:"\f251"}.fa-house-damage:before{content:"\f6f1"}.fa-house-user:before{content:"\e065"}.fa-houzz:before{content:"\f27c"}.fa-hryvnia:before{content:"\f6f2"}.fa-html5:before{content:"\f13b"}.fa-hubspot:before{content:"\f3b2"}.fa-i-cursor:before{content:"\f246"}.fa-ice-cream:before{content:"\f810"}.fa-icicles:before{content:"\f7ad"}.fa-icons:before{content:"\f86d"}.fa-id-badge:before{content:"\f2c1"}.fa-id-card:before{content:"\f2c2"}.fa-id-card-alt:before{content:"\f47f"}.fa-ideal:before{content:"\e013"}.fa-igloo:before{content:"\f7ae"}.fa-image:before{content:"\f03e"}.fa-images:before{content:"\f302"}.fa-imdb:before{content:"\f2d8"}.fa-inbox:before{content:"\f01c"}.fa-indent:before{content:"\f03c"}.fa-industry:before{content:"\f275"}.fa-infinity:before{content:"\f534"}.fa-info:before{content:"\f129"}.fa-info-circle:before{content:"\f05a"}.fa-innosoft:before{content:"\e080"}.fa-instagram:before{content:"\f16d"}.fa-instagram-square:before{content:"\e055"}.fa-instalod:before{content:"\e081"}.fa-intercom:before{content:"\f7af"}.fa-internet-explorer:before{content:"\f26b"}.fa-invision:before{content:"\f7b0"}.fa-ioxhost:before{content:"\f208"}.fa-italic:before{content:"\f033"}.fa-itch-io:before{content:"\f83a"}.fa-itunes:before{content:"\f3b4"}.fa-itunes-note:before{content:"\f3b5"}.fa-java:before{content:"\f4e4"}.fa-jedi:before{content:"\f669"}.fa-jedi-order:before{content:"\f50e"}.fa-jenkins:before{content:"\f3b6"}.fa-jira:before{content:"\f7b1"}.fa-joget:before{content:"\f3b7"}.fa-joint:before{content:"\f595"}.fa-joomla:before{content:"\f1aa"}.fa-journal-whills:before{content:"\f66a"}.fa-js:before{content:"\f3b8"}.fa-js-square:before{content:"\f3b9"}.fa-jsfiddle:before{content:"\f1cc"}.fa-kaaba:before{content:"\f66b"}.fa-kaggle:before{content:"\f5fa"}.fa-key:before{content:"\f084"}.fa-keybase:before{content:"\f4f5"}.fa-keyboard:before{content:"\f11c"}.fa-keycdn:before{content:"\f3ba"}.fa-khanda:before{content:"\f66d"}.fa-kickstarter:before{content:"\f3bb"}.fa-kickstarter-k:before{content:"\f3bc"}.fa-kiss:before{content:"\f596"}.fa-kiss-beam:before{content:"\f597"}.fa-kiss-wink-heart:before{content:"\f598"}.fa-kiwi-bird:before{content:"\f535"}.fa-korvue:before{content:"\f42f"}.fa-landmark:before{content:"\f66f"}.fa-language:before{content:"\f1ab"}.fa-laptop:before{content:"\f109"}.fa-laptop-code:before{content:"\f5fc"}.fa-laptop-house:before{content:"\e066"}.fa-laptop-medical:before{content:"\f812"}.fa-laravel:before{content:"\f3bd"}.fa-lastfm:before{content:"\f202"}.fa-lastfm-square:before{content:"\f203"}.fa-laugh:before{content:"\f599"}.fa-laugh-beam:before{content:"\f59a"}.fa-laugh-squint:before{content:"\f59b"}.fa-laugh-wink:before{content:"\f59c"}.fa-layer-group:before{content:"\f5fd"}.fa-leaf:before{content:"\f06c"}.fa-leanpub:before{content:"\f212"}.fa-lemon:before{content:"\f094"}.fa-less:before{content:"\f41d"}.fa-less-than:before{content:"\f536"}.fa-less-than-equal:before{content:"\f537"}.fa-level-down-alt:before{content:"\f3be"}.fa-level-up-alt:before{content:"\f3bf"}.fa-life-ring:before{content:"\f1cd"}.fa-lightbulb:before{content:"\f0eb"}.fa-line:before{content:"\f3c0"}.fa-link:before{content:"\f0c1"}.fa-linkedin:before{content:"\f08c"}.fa-linkedin-in:before{content:"\f0e1"}.fa-linode:before{content:"\f2b8"}.fa-linux:before{content:"\f17c"}.fa-lira-sign:before{content:"\f195"}.fa-list:before{content:"\f03a"}.fa-list-alt:before{content:"\f022"}.fa-list-ol:before{content:"\f0cb"}.fa-list-ul:before{content:"\f0ca"}.fa-location-arrow:before{content:"\f124"}.fa-lock:before{content:"\f023"}.fa-lock-open:before{content:"\f3c1"}.fa-long-arrow-alt-down:before{content:"\f309"}.fa-long-arrow-alt-left:before{content:"\f30a"}.fa-long-arrow-alt-right:before{content:"\f30b"}.fa-long-arrow-alt-up:before{content:"\f30c"}.fa-low-vision:before{content:"\f2a8"}.fa-luggage-cart:before{content:"\f59d"}.fa-lungs:before{content:"\f604"}.fa-lungs-virus:before{content:"\e067"}.fa-lyft:before{content:"\f3c3"}.fa-magento:before{content:"\f3c4"}.fa-magic:before{content:"\f0d0"}.fa-magnet:before{content:"\f076"}.fa-mail-bulk:before{content:"\f674"}.fa-mailchimp:before{content:"\f59e"}.fa-male:before{content:"\f183"}.fa-mandalorian:before{content:"\f50f"}.fa-map:before{content:"\f279"}.fa-map-marked:before{content:"\f59f"}.fa-map-marked-alt:before{content:"\f5a0"}.fa-map-marker:before{content:"\f041"}.fa-map-marker-alt:before{content:"\f3c5"}.fa-map-pin:before{content:"\f276"}.fa-map-signs:before{content:"\f277"}.fa-markdown:before{content:"\f60f"}.fa-marker:before{content:"\f5a1"}.fa-mars:before{content:"\f222"}.fa-mars-double:before{content:"\f227"}.fa-mars-stroke:before{content:"\f229"}.fa-mars-stroke-h:before{content:"\f22b"}.fa-mars-stroke-v:before{content:"\f22a"}.fa-mask:before{content:"\f6fa"}.fa-mastodon:before{content:"\f4f6"}.fa-maxcdn:before{content:"\f136"}.fa-mdb:before{content:"\f8ca"}.fa-medal:before{content:"\f5a2"}.fa-medapps:before{content:"\f3c6"}.fa-medium:before{content:"\f23a"}.fa-medium-m:before{content:"\f3c7"}.fa-medkit:before{content:"\f0fa"}.fa-medrt:before{content:"\f3c8"}.fa-meetup:before{content:"\f2e0"}.fa-megaport:before{content:"\f5a3"}.fa-meh:before{content:"\f11a"}.fa-meh-blank:before{content:"\f5a4"}.fa-meh-rolling-eyes:before{content:"\f5a5"}.fa-memory:before{content:"\f538"}.fa-mendeley:before{content:"\f7b3"}.fa-menorah:before{content:"\f676"}.fa-mercury:before{content:"\f223"}.fa-meteor:before{content:"\f753"}.fa-microblog:before{content:"\e01a"}.fa-microchip:before{content:"\f2db"}.fa-microphone:before{content:"\f130"}.fa-microphone-alt:before{content:"\f3c9"}.fa-microphone-alt-slash:before{content:"\f539"}.fa-microphone-slash:before{content:"\f131"}.fa-microscope:before{content:"\f610"}.fa-microsoft:before{content:"\f3ca"}.fa-minus:before{content:"\f068"}.fa-minus-circle:before{content:"\f056"}.fa-minus-square:before{content:"\f146"}.fa-mitten:before{content:"\f7b5"}.fa-mix:before{content:"\f3cb"}.fa-mixcloud:before{content:"\f289"}.fa-mixer:before{content:"\e056"}.fa-mizuni:before{content:"\f3cc"}.fa-mobile:before{content:"\f10b"}.fa-mobile-alt:before{content:"\f3cd"}.fa-modx:before{content:"\f285"}.fa-monero:before{content:"\f3d0"}.fa-money-bill:before{content:"\f0d6"}.fa-money-bill-alt:before{content:"\f3d1"}.fa-money-bill-wave:before{content:"\f53a"}.fa-money-bill-wave-alt:before{content:"\f53b"}.fa-money-check:before{content:"\f53c"}.fa-money-check-alt:before{content:"\f53d"}.fa-monument:before{content:"\f5a6"}.fa-moon:before{content:"\f186"}.fa-mortar-pestle:before{content:"\f5a7"}.fa-mosque:before{content:"\f678"}.fa-motorcycle:before{content:"\f21c"}.fa-mountain:before{content:"\f6fc"}.fa-mouse:before{content:"\f8cc"}.fa-mouse-pointer:before{content:"\f245"}.fa-mug-hot:before{content:"\f7b6"}.fa-music:before{content:"\f001"}.fa-napster:before{content:"\f3d2"}.fa-neos:before{content:"\f612"}.fa-network-wired:before{content:"\f6ff"}.fa-neuter:before{content:"\f22c"}.fa-newspaper:before{content:"\f1ea"}.fa-nimblr:before{content:"\f5a8"}.fa-node:before{content:"\f419"}.fa-node-js:before{content:"\f3d3"}.fa-not-equal:before{content:"\f53e"}.fa-notes-medical:before{content:"\f481"}.fa-npm:before{content:"\f3d4"}.fa-ns8:before{content:"\f3d5"}.fa-nutritionix:before{content:"\f3d6"}.fa-object-group:before{content:"\f247"}.fa-object-ungroup:before{content:"\f248"}.fa-octopus-deploy:before{content:"\e082"}.fa-odnoklassniki:before{content:"\f263"}.fa-odnoklassniki-square:before{content:"\f264"}.fa-oil-can:before{content:"\f613"}.fa-old-republic:before{content:"\f510"}.fa-om:before{content:"\f679"}.fa-opencart:before{content:"\f23d"}.fa-openid:before{content:"\f19b"}.fa-opera:before{content:"\f26a"}.fa-optin-monster:before{content:"\f23c"}.fa-orcid:before{content:"\f8d2"}.fa-osi:before{content:"\f41a"}.fa-otter:before{content:"\f700"}.fa-outdent:before{content:"\f03b"}.fa-page4:before{content:"\f3d7"}.fa-pagelines:before{content:"\f18c"}.fa-pager:before{content:"\f815"}.fa-paint-brush:before{content:"\f1fc"}.fa-paint-roller:before{content:"\f5aa"}.fa-palette:before{content:"\f53f"}.fa-palfed:before{content:"\f3d8"}.fa-pallet:before{content:"\f482"}.fa-paper-plane:before{content:"\f1d8"}.fa-paperclip:before{content:"\f0c6"}.fa-parachute-box:before{content:"\f4cd"}.fa-paragraph:before{content:"\f1dd"}.fa-parking:before{content:"\f540"}.fa-passport:before{content:"\f5ab"}.fa-pastafarianism:before{content:"\f67b"}.fa-paste:before{content:"\f0ea"}.fa-patreon:before{content:"\f3d9"}.fa-pause:before{content:"\f04c"}.fa-pause-circle:before{content:"\f28b"}.fa-paw:before{content:"\f1b0"}.fa-paypal:before{content:"\f1ed"}.fa-peace:before{content:"\f67c"}.fa-pen:before{content:"\f304"}.fa-pen-alt:before{content:"\f305"}.fa-pen-fancy:before{content:"\f5ac"}.fa-pen-nib:before{content:"\f5ad"}.fa-pen-square:before{content:"\f14b"}.fa-pencil-alt:before{content:"\f303"}.fa-pencil-ruler:before{content:"\f5ae"}.fa-penny-arcade:before{content:"\f704"}.fa-people-arrows:before{content:"\e068"}.fa-people-carry:before{content:"\f4ce"}.fa-pepper-hot:before{content:"\f816"}.fa-perbyte:before{content:"\e083"}.fa-percent:before{content:"\f295"}.fa-percentage:before{content:"\f541"}.fa-periscope:before{content:"\f3da"}.fa-person-booth:before{content:"\f756"}.fa-phabricator:before{content:"\f3db"}.fa-phoenix-framework:before{content:"\f3dc"}.fa-phoenix-squadron:before{content:"\f511"}.fa-phone:before{content:"\f095"}.fa-phone-alt:before{content:"\f879"}.fa-phone-slash:before{content:"\f3dd"}.fa-phone-square:before{content:"\f098"}.fa-phone-square-alt:before{content:"\f87b"}.fa-phone-volume:before{content:"\f2a0"}.fa-photo-video:before{content:"\f87c"}.fa-php:before{content:"\f457"}.fa-pied-piper:before{content:"\f2ae"}.fa-pied-piper-alt:before{content:"\f1a8"}.fa-pied-piper-hat:before{content:"\f4e5"}.fa-pied-piper-pp:before{content:"\f1a7"}.fa-pied-piper-square:before{content:"\e01e"}.fa-piggy-bank:before{content:"\f4d3"}.fa-pills:before{content:"\f484"}.fa-pinterest:before{content:"\f0d2"}.fa-pinterest-p:before{content:"\f231"}.fa-pinterest-square:before{content:"\f0d3"}.fa-pizza-slice:before{content:"\f818"}.fa-place-of-worship:before{content:"\f67f"}.fa-plane:before{content:"\f072"}.fa-plane-arrival:before{content:"\f5af"}.fa-plane-departure:before{content:"\f5b0"}.fa-plane-slash:before{content:"\e069"}.fa-play:before{content:"\f04b"}.fa-play-circle:before{content:"\f144"}.fa-playstation:before{content:"\f3df"}.fa-plug:before{content:"\f1e6"}.fa-plus:before{content:"\f067"}.fa-plus-circle:before{content:"\f055"}.fa-plus-square:before{content:"\f0fe"}.fa-podcast:before{content:"\f2ce"}.fa-poll:before{content:"\f681"}.fa-poll-h:before{content:"\f682"}.fa-poo:before{content:"\f2fe"}.fa-poo-storm:before{content:"\f75a"}.fa-poop:before{content:"\f619"}.fa-portrait:before{content:"\f3e0"}.fa-pound-sign:before{content:"\f154"}.fa-power-off:before{content:"\f011"}.fa-pray:before{content:"\f683"}.fa-praying-hands:before{content:"\f684"}.fa-prescription:before{content:"\f5b1"}.fa-prescription-bottle:before{content:"\f485"}.fa-prescription-bottle-alt:before{content:"\f486"}.fa-print:before{content:"\f02f"}.fa-procedures:before{content:"\f487"}.fa-product-hunt:before{content:"\f288"}.fa-project-diagram:before{content:"\f542"}.fa-pump-medical:before{content:"\e06a"}.fa-pump-soap:before{content:"\e06b"}.fa-pushed:before{content:"\f3e1"}.fa-puzzle-piece:before{content:"\f12e"}.fa-python:before{content:"\f3e2"}.fa-qq:before{content:"\f1d6"}.fa-qrcode:before{content:"\f029"}.fa-question:before{content:"\f128"}.fa-question-circle:before{content:"\f059"}.fa-quidditch:before{content:"\f458"}.fa-quinscape:before{content:"\f459"}.fa-quora:before{content:"\f2c4"}.fa-quote-left:before{content:"\f10d"}.fa-quote-right:before{content:"\f10e"}.fa-quran:before{content:"\f687"}.fa-r-project:before{content:"\f4f7"}.fa-radiation:before{content:"\f7b9"}.fa-radiation-alt:before{content:"\f7ba"}.fa-rainbow:before{content:"\f75b"}.fa-random:before{content:"\f074"}.fa-raspberry-pi:before{content:"\f7bb"}.fa-ravelry:before{content:"\f2d9"}.fa-react:before{content:"\f41b"}.fa-reacteurope:before{content:"\f75d"}.fa-readme:before{content:"\f4d5"}.fa-rebel:before{content:"\f1d0"}.fa-receipt:before{content:"\f543"}.fa-record-vinyl:before{content:"\f8d9"}.fa-recycle:before{content:"\f1b8"}.fa-red-river:before{content:"\f3e3"}.fa-reddit:before{content:"\f1a1"}.fa-reddit-alien:before{content:"\f281"}.fa-reddit-square:before{content:"\f1a2"}.fa-redhat:before{content:"\f7bc"}.fa-redo:before{content:"\f01e"}.fa-redo-alt:before{content:"\f2f9"}.fa-registered:before{content:"\f25d"}.fa-remove-format:before{content:"\f87d"}.fa-renren:before{content:"\f18b"}.fa-reply:before{content:"\f3e5"}.fa-reply-all:before{content:"\f122"}.fa-replyd:before{content:"\f3e6"}.fa-republican:before{content:"\f75e"}.fa-researchgate:before{content:"\f4f8"}.fa-resolving:before{content:"\f3e7"}.fa-restroom:before{content:"\f7bd"}.fa-retweet:before{content:"\f079"}.fa-rev:before{content:"\f5b2"}.fa-ribbon:before{content:"\f4d6"}.fa-ring:before{content:"\f70b"}.fa-road:before{content:"\f018"}.fa-robot:before{content:"\f544"}.fa-rocket:before{content:"\f135"}.fa-rocketchat:before{content:"\f3e8"}.fa-rockrms:before{content:"\f3e9"}.fa-route:before{content:"\f4d7"}.fa-rss:before{content:"\f09e"}.fa-rss-square:before{content:"\f143"}.fa-ruble-sign:before{content:"\f158"}.fa-ruler:before{content:"\f545"}.fa-ruler-combined:before{content:"\f546"}.fa-ruler-horizontal:before{content:"\f547"}.fa-ruler-vertical:before{content:"\f548"}.fa-running:before{content:"\f70c"}.fa-rupee-sign:before{content:"\f156"}.fa-rust:before{content:"\e07a"}.fa-sad-cry:before{content:"\f5b3"}.fa-sad-tear:before{content:"\f5b4"}.fa-safari:before{content:"\f267"}.fa-salesforce:before{content:"\f83b"}.fa-sass:before{content:"\f41e"}.fa-satellite:before{content:"\f7bf"}.fa-satellite-dish:before{content:"\f7c0"}.fa-save:before{content:"\f0c7"}.fa-schlix:before{content:"\f3ea"}.fa-school:before{content:"\f549"}.fa-screwdriver:before{content:"\f54a"}.fa-scribd:before{content:"\f28a"}.fa-scroll:before{content:"\f70e"}.fa-sd-card:before{content:"\f7c2"}.fa-search:before{content:"\f002"}.fa-search-dollar:before{content:"\f688"}.fa-search-location:before{content:"\f689"}.fa-search-minus:before{content:"\f010"}.fa-search-plus:before{content:"\f00e"}.fa-searchengin:before{content:"\f3eb"}.fa-seedling:before{content:"\f4d8"}.fa-sellcast:before{content:"\f2da"}.fa-sellsy:before{content:"\f213"}.fa-server:before{content:"\f233"}.fa-servicestack:before{content:"\f3ec"}.fa-shapes:before{content:"\f61f"}.fa-share:before{content:"\f064"}.fa-share-alt:before{content:"\f1e0"}.fa-share-alt-square:before{content:"\f1e1"}.fa-share-square:before{content:"\f14d"}.fa-shekel-sign:before{content:"\f20b"}.fa-shield-alt:before{content:"\f3ed"}.fa-shield-virus:before{content:"\e06c"}.fa-ship:before{content:"\f21a"}.fa-shipping-fast:before{content:"\f48b"}.fa-shirtsinbulk:before{content:"\f214"}.fa-shoe-prints:before{content:"\f54b"}.fa-shopify:before{content:"\e057"}.fa-shopping-bag:before{content:"\f290"}.fa-shopping-basket:before{content:"\f291"}.fa-shopping-cart:before{content:"\f07a"}.fa-shopware:before{content:"\f5b5"}.fa-shower:before{content:"\f2cc"}.fa-shuttle-van:before{content:"\f5b6"}.fa-sign:before{content:"\f4d9"}.fa-sign-in-alt:before{content:"\f2f6"}.fa-sign-language:before{content:"\f2a7"}.fa-sign-out-alt:before{content:"\f2f5"}.fa-signal:before{content:"\f012"}.fa-signature:before{content:"\f5b7"}.fa-sim-card:before{content:"\f7c4"}.fa-simplybuilt:before{content:"\f215"}.fa-sink:before{content:"\e06d"}.fa-sistrix:before{content:"\f3ee"}.fa-sitemap:before{content:"\f0e8"}.fa-sith:before{content:"\f512"}.fa-skating:before{content:"\f7c5"}.fa-sketch:before{content:"\f7c6"}.fa-skiing:before{content:"\f7c9"}.fa-skiing-nordic:before{content:"\f7ca"}.fa-skull:before{content:"\f54c"}.fa-skull-crossbones:before{content:"\f714"}.fa-skyatlas:before{content:"\f216"}.fa-skype:before{content:"\f17e"}.fa-slack:before{content:"\f198"}.fa-slack-hash:before{content:"\f3ef"}.fa-slash:before{content:"\f715"}.fa-sleigh:before{content:"\f7cc"}.fa-sliders-h:before{content:"\f1de"}.fa-slideshare:before{content:"\f1e7"}.fa-smile:before{content:"\f118"}.fa-smile-beam:before{content:"\f5b8"}.fa-smile-wink:before{content:"\f4da"}.fa-smog:before{content:"\f75f"}.fa-smoking:before{content:"\f48d"}.fa-smoking-ban:before{content:"\f54d"}.fa-sms:before{content:"\f7cd"}.fa-snapchat:before{content:"\f2ab"}.fa-snapchat-ghost:before{content:"\f2ac"}.fa-snapchat-square:before{content:"\f2ad"}.fa-snowboarding:before{content:"\f7ce"}.fa-snowflake:before{content:"\f2dc"}.fa-snowman:before{content:"\f7d0"}.fa-snowplow:before{content:"\f7d2"}.fa-soap:before{content:"\e06e"}.fa-socks:before{content:"\f696"}.fa-solar-panel:before{content:"\f5ba"}.fa-sort:before{content:"\f0dc"}.fa-sort-alpha-down:before{content:"\f15d"}.fa-sort-alpha-down-alt:before{content:"\f881"}.fa-sort-alpha-up:before{content:"\f15e"}.fa-sort-alpha-up-alt:before{content:"\f882"}.fa-sort-amount-down:before{content:"\f160"}.fa-sort-amount-down-alt:before{content:"\f884"}.fa-sort-amount-up:before{content:"\f161"}.fa-sort-amount-up-alt:before{content:"\f885"}.fa-sort-down:before{content:"\f0dd"}.fa-sort-numeric-down:before{content:"\f162"}.fa-sort-numeric-down-alt:before{content:"\f886"}.fa-sort-numeric-up:before{content:"\f163"}.fa-sort-numeric-up-alt:before{content:"\f887"}.fa-sort-up:before{content:"\f0de"}.fa-soundcloud:before{content:"\f1be"}.fa-sourcetree:before{content:"\f7d3"}.fa-spa:before{content:"\f5bb"}.fa-space-shuttle:before{content:"\f197"}.fa-speakap:before{content:"\f3f3"}.fa-speaker-deck:before{content:"\f83c"}.fa-spell-check:before{content:"\f891"}.fa-spider:before{content:"\f717"}.fa-spinner:before{content:"\f110"}.fa-splotch:before{content:"\f5bc"}.fa-spotify:before{content:"\f1bc"}.fa-spray-can:before{content:"\f5bd"}.fa-square:before{content:"\f0c8"}.fa-square-full:before{content:"\f45c"}.fa-square-root-alt:before{content:"\f698"}.fa-squarespace:before{content:"\f5be"}.fa-stack-exchange:before{content:"\f18d"}.fa-stack-overflow:before{content:"\f16c"}.fa-stackpath:before{content:"\f842"}.fa-stamp:before{content:"\f5bf"}.fa-star:before{content:"\f005"}.fa-star-and-crescent:before{content:"\f699"}.fa-star-half:before{content:"\f089"}.fa-star-half-alt:before{content:"\f5c0"}.fa-star-of-david:before{content:"\f69a"}.fa-star-of-life:before{content:"\f621"}.fa-staylinked:before{content:"\f3f5"}.fa-steam:before{content:"\f1b6"}.fa-steam-square:before{content:"\f1b7"}.fa-steam-symbol:before{content:"\f3f6"}.fa-step-backward:before{content:"\f048"}.fa-step-forward:before{content:"\f051"}.fa-stethoscope:before{content:"\f0f1"}.fa-sticker-mule:before{content:"\f3f7"}.fa-sticky-note:before{content:"\f249"}.fa-stop:before{content:"\f04d"}.fa-stop-circle:before{content:"\f28d"}.fa-stopwatch:before{content:"\f2f2"}.fa-stopwatch-20:before{content:"\e06f"}.fa-store:before{content:"\f54e"}.fa-store-alt:before{content:"\f54f"}.fa-store-alt-slash:before{content:"\e070"}.fa-store-slash:before{content:"\e071"}.fa-strava:before{content:"\f428"}.fa-stream:before{content:"\f550"}.fa-street-view:before{content:"\f21d"}.fa-strikethrough:before{content:"\f0cc"}.fa-stripe:before{content:"\f429"}.fa-stripe-s:before{content:"\f42a"}.fa-stroopwafel:before{content:"\f551"}.fa-studiovinari:before{content:"\f3f8"}.fa-stumbleupon:before{content:"\f1a4"}.fa-stumbleupon-circle:before{content:"\f1a3"}.fa-subscript:before{content:"\f12c"}.fa-subway:before{content:"\f239"}.fa-suitcase:before{content:"\f0f2"}.fa-suitcase-rolling:before{content:"\f5c1"}.fa-sun:before{content:"\f185"}.fa-superpowers:before{content:"\f2dd"}.fa-superscript:before{content:"\f12b"}.fa-supple:before{content:"\f3f9"}.fa-surprise:before{content:"\f5c2"}.fa-suse:before{content:"\f7d6"}.fa-swatchbook:before{content:"\f5c3"}.fa-swift:before{content:"\f8e1"}.fa-swimmer:before{content:"\f5c4"}.fa-swimming-pool:before{content:"\f5c5"}.fa-symfony:before{content:"\f83d"}.fa-synagogue:before{content:"\f69b"}.fa-sync:before{content:"\f021"}.fa-sync-alt:before{content:"\f2f1"}.fa-syringe:before{content:"\f48e"}.fa-table:before{content:"\f0ce"}.fa-table-tennis:before{content:"\f45d"}.fa-tablet:before{content:"\f10a"}.fa-tablet-alt:before{content:"\f3fa"}.fa-tablets:before{content:"\f490"}.fa-tachometer-alt:before{content:"\f3fd"}.fa-tag:before{content:"\f02b"}.fa-tags:before{content:"\f02c"}.fa-tape:before{content:"\f4db"}.fa-tasks:before{content:"\f0ae"}.fa-taxi:before{content:"\f1ba"}.fa-teamspeak:before{content:"\f4f9"}.fa-teeth:before{content:"\f62e"}.fa-teeth-open:before{content:"\f62f"}.fa-telegram:before{content:"\f2c6"}.fa-telegram-plane:before{content:"\f3fe"}.fa-temperature-high:before{content:"\f769"}.fa-temperature-low:before{content:"\f76b"}.fa-tencent-weibo:before{content:"\f1d5"}.fa-tenge:before{content:"\f7d7"}.fa-terminal:before{content:"\f120"}.fa-text-height:before{content:"\f034"}.fa-text-width:before{content:"\f035"}.fa-th:before{content:"\f00a"}.fa-th-large:before{content:"\f009"}.fa-th-list:before{content:"\f00b"}.fa-the-red-yeti:before{content:"\f69d"}.fa-theater-masks:before{content:"\f630"}.fa-themeco:before{content:"\f5c6"}.fa-themeisle:before{content:"\f2b2"}.fa-thermometer:before{content:"\f491"}.fa-thermometer-empty:before{content:"\f2cb"}.fa-thermometer-full:before{content:"\f2c7"}.fa-thermometer-half:before{content:"\f2c9"}.fa-thermometer-quarter:before{content:"\f2ca"}.fa-thermometer-three-quarters:before{content:"\f2c8"}.fa-think-peaks:before{content:"\f731"}.fa-thumbs-down:before{content:"\f165"}.fa-thumbs-up:before{content:"\f164"}.fa-thumbtack:before{content:"\f08d"}.fa-ticket-alt:before{content:"\f3ff"}.fa-tiktok:before{content:"\e07b"}.fa-times:before{content:"\f00d"}.fa-times-circle:before{content:"\f057"}.fa-tint:before{content:"\f043"}.fa-tint-slash:before{content:"\f5c7"}.fa-tired:before{content:"\f5c8"}.fa-toggle-off:before{content:"\f204"}.fa-toggle-on:before{content:"\f205"}.fa-toilet:before{content:"\f7d8"}.fa-toilet-paper:before{content:"\f71e"}.fa-toilet-paper-slash:before{content:"\e072"}.fa-toolbox:before{content:"\f552"}.fa-tools:before{content:"\f7d9"}.fa-tooth:before{content:"\f5c9"}.fa-torah:before{content:"\f6a0"}.fa-torii-gate:before{content:"\f6a1"}.fa-tractor:before{content:"\f722"}.fa-trade-federation:before{content:"\f513"}.fa-trademark:before{content:"\f25c"}.fa-traffic-light:before{content:"\f637"}.fa-trailer:before{content:"\e041"}.fa-train:before{content:"\f238"}.fa-tram:before{content:"\f7da"}.fa-transgender:before{content:"\f224"}.fa-transgender-alt:before{content:"\f225"}.fa-trash:before{content:"\f1f8"}.fa-trash-alt:before{content:"\f2ed"}.fa-trash-restore:before{content:"\f829"}.fa-trash-restore-alt:before{content:"\f82a"}.fa-tree:before{content:"\f1bb"}.fa-trello:before{content:"\f181"}.fa-trophy:before{content:"\f091"}.fa-truck:before{content:"\f0d1"}.fa-truck-loading:before{content:"\f4de"}.fa-truck-monster:before{content:"\f63b"}.fa-truck-moving:before{content:"\f4df"}.fa-truck-pickup:before{content:"\f63c"}.fa-tshirt:before{content:"\f553"}.fa-tty:before{content:"\f1e4"}.fa-tumblr:before{content:"\f173"}.fa-tumblr-square:before{content:"\f174"}.fa-tv:before{content:"\f26c"}.fa-twitch:before{content:"\f1e8"}.fa-twitter:before{content:"\f099"}.fa-twitter-square:before{content:"\f081"}.fa-typo3:before{content:"\f42b"}.fa-uber:before{content:"\f402"}.fa-ubuntu:before{content:"\f7df"}.fa-uikit:before{content:"\f403"}.fa-umbraco:before{content:"\f8e8"}.fa-umbrella:before{content:"\f0e9"}.fa-umbrella-beach:before{content:"\f5ca"}.fa-uncharted:before{content:"\e084"}.fa-underline:before{content:"\f0cd"}.fa-undo:before{content:"\f0e2"}.fa-undo-alt:before{content:"\f2ea"}.fa-uniregistry:before{content:"\f404"}.fa-unity:before{content:"\e049"}.fa-universal-access:before{content:"\f29a"}.fa-university:before{content:"\f19c"}.fa-unlink:before{content:"\f127"}.fa-unlock:before{content:"\f09c"}.fa-unlock-alt:before{content:"\f13e"}.fa-unsplash:before{content:"\e07c"}.fa-untappd:before{content:"\f405"}.fa-upload:before{content:"\f093"}.fa-ups:before{content:"\f7e0"}.fa-usb:before{content:"\f287"}.fa-user:before{content:"\f007"}.fa-user-alt:before{content:"\f406"}.fa-user-alt-slash:before{content:"\f4fa"}.fa-user-astronaut:before{content:"\f4fb"}.fa-user-check:before{content:"\f4fc"}.fa-user-circle:before{content:"\f2bd"}.fa-user-clock:before{content:"\f4fd"}.fa-user-cog:before{content:"\f4fe"}.fa-user-edit:before{content:"\f4ff"}.fa-user-friends:before{content:"\f500"}.fa-user-graduate:before{content:"\f501"}.fa-user-injured:before{content:"\f728"}.fa-user-lock:before{content:"\f502"}.fa-user-md:before{content:"\f0f0"}.fa-user-minus:before{content:"\f503"}.fa-user-ninja:before{content:"\f504"}.fa-user-nurse:before{content:"\f82f"}.fa-user-plus:before{content:"\f234"}.fa-user-secret:before{content:"\f21b"}.fa-user-shield:before{content:"\f505"}.fa-user-slash:before{content:"\f506"}.fa-user-tag:before{content:"\f507"}.fa-user-tie:before{content:"\f508"}.fa-user-times:before{content:"\f235"}.fa-users:before{content:"\f0c0"}.fa-users-cog:before{content:"\f509"}.fa-users-slash:before{content:"\e073"}.fa-usps:before{content:"\f7e1"}.fa-ussunnah:before{content:"\f407"}.fa-utensil-spoon:before{content:"\f2e5"}.fa-utensils:before{content:"\f2e7"}.fa-vaadin:before{content:"\f408"}.fa-vector-square:before{content:"\f5cb"}.fa-venus:before{content:"\f221"}.fa-venus-double:before{content:"\f226"}.fa-venus-mars:before{content:"\f228"}.fa-vest:before{content:"\e085"}.fa-vest-patches:before{content:"\e086"}.fa-viacoin:before{content:"\f237"}.fa-viadeo:before{content:"\f2a9"}.fa-viadeo-square:before{content:"\f2aa"}.fa-vial:before{content:"\f492"}.fa-vials:before{content:"\f493"}.fa-viber:before{content:"\f409"}.fa-video:before{content:"\f03d"}.fa-video-slash:before{content:"\f4e2"}.fa-vihara:before{content:"\f6a7"}.fa-vimeo:before{content:"\f40a"}.fa-vimeo-square:before{content:"\f194"}.fa-vimeo-v:before{content:"\f27d"}.fa-vine:before{content:"\f1ca"}.fa-virus:before{content:"\e074"}.fa-virus-slash:before{content:"\e075"}.fa-viruses:before{content:"\e076"}.fa-vk:before{content:"\f189"}.fa-vnv:before{content:"\f40b"}.fa-voicemail:before{content:"\f897"}.fa-volleyball-ball:before{content:"\f45f"}.fa-volume-down:before{content:"\f027"}.fa-volume-mute:before{content:"\f6a9"}.fa-volume-off:before{content:"\f026"}.fa-volume-up:before{content:"\f028"}.fa-vote-yea:before{content:"\f772"}.fa-vr-cardboard:before{content:"\f729"}.fa-vuejs:before{content:"\f41f"}.fa-walking:before{content:"\f554"}.fa-wallet:before{content:"\f555"}.fa-warehouse:before{content:"\f494"}.fa-watchman-monitoring:before{content:"\e087"}.fa-water:before{content:"\f773"}.fa-wave-square:before{content:"\f83e"}.fa-waze:before{content:"\f83f"}.fa-weebly:before{content:"\f5cc"}.fa-weibo:before{content:"\f18a"}.fa-weight:before{content:"\f496"}.fa-weight-hanging:before{content:"\f5cd"}.fa-weixin:before{content:"\f1d7"}.fa-whatsapp:before{content:"\f232"}.fa-whatsapp-square:before{content:"\f40c"}.fa-wheelchair:before{content:"\f193"}.fa-whmcs:before{content:"\f40d"}.fa-wifi:before{content:"\f1eb"}.fa-wikipedia-w:before{content:"\f266"}.fa-wind:before{content:"\f72e"}.fa-window-close:before{content:"\f410"}.fa-window-maximize:before{content:"\f2d0"}.fa-window-minimize:before{content:"\f2d1"}.fa-window-restore:before{content:"\f2d2"}.fa-windows:before{content:"\f17a"}.fa-wine-bottle:before{content:"\f72f"}.fa-wine-glass:before{content:"\f4e3"}.fa-wine-glass-alt:before{content:"\f5ce"}.fa-wix:before{content:"\f5cf"}.fa-wizards-of-the-coast:before{content:"\f730"}.fa-wodu:before{content:"\e088"}.fa-wolf-pack-battalion:before{content:"\f514"}.fa-won-sign:before{content:"\f159"}.fa-wordpress:before{content:"\f19a"}.fa-wordpress-simple:before{content:"\f411"}.fa-wpbeginner:before{content:"\f297"}.fa-wpexplorer:before{content:"\f2de"}.fa-wpforms:before{content:"\f298"}.fa-wpressr:before{content:"\f3e4"}.fa-wrench:before{content:"\f0ad"}.fa-x-ray:before{content:"\f497"}.fa-xbox:before{content:"\f412"}.fa-xing:before{content:"\f168"}.fa-xing-square:before{content:"\f169"}.fa-y-combinator:before{content:"\f23b"}.fa-yahoo:before{content:"\f19e"}.fa-yammer:before{content:"\f840"}.fa-yandex:before{content:"\f413"}.fa-yandex-international:before{content:"\f414"}.fa-yarn:before{content:"\f7e3"}.fa-yelp:before{content:"\f1e9"}.fa-yen-sign:before{content:"\f157"}.fa-yin-yang:before{content:"\f6ad"}.fa-yoast:before{content:"\f2b1"}.fa-youtube:before{content:"\f167"}.fa-youtube-square:before{content:"\f431"}.fa-zhihu:before{content:"\f63f"}.sr-only{border:0;clip:rect(0,0,0,0);height:1px;margin:-1px;overflow:hidden;padding:0;position:absolute;width:1px}.sr-only-focusable:active,.sr-only-focusable:focus{clip:auto;height:auto;margin:0;overflow:visible;position:static;width:auto}@font-face{font-family:"Font Awesome 5 Brands";font-style:normal;font-weight:400;font-display:block;src:url(webfonts/fa-brands-400.eot);src:url(webfonts/fa-brands-400.eot?#iefix) format("embedded-opentype"),url(webfonts/fa-brands-400.woff2) format("woff2"),url(webfonts/fa-brands-400.woff) format("woff"),url(webfonts/fa-brands-400.ttf) format("truetype"),url(webfonts/fa-brands-400.svg#fontawesome) format("svg")}.fab{font-family:"Font Awesome 5 Brands"}@font-face{font-family:"Font Awesome 5 Free";font-style:normal;font-weight:400;font-display:block;src:url(webfonts/fa-regular-400.eot);src:url(webfonts/fa-regular-400.eot?#iefix) format("embedded-opentype"),url(webfonts/fa-regular-400.woff2) format("woff2"),url(webfonts/fa-regular-400.woff) format("woff"),url(webfonts/fa-regular-400.ttf) format("truetype"),url(webfonts/fa-regular-400.svg#fontawesome) format("svg")}.fab,.far{font-weight:400}@font-face{font-family:"Font Awesome 5 Free";font-style:normal;font-weight:900;font-display:block;src:url(webfonts/fa-solid-900.eot);src:url(webfonts/fa-solid-900.eot?#iefix) format("embedded-opentype"),url(webfonts/fa-solid-900.woff2) format("woff2"),url(webfonts/fa-solid-900.woff) format("woff"),url(webfonts/fa-solid-900.ttf) format("truetype"),url(webfonts/fa-solid-900.svg#fontawesome) format("svg")}.fa,.far,.fas{font-family:"Font Awesome 5 Free"}.fa,.fas{font-weight:900}
--------------------------------------------------------------------------------