feat: 完善项目。
- 后端代码结构调整; - 所有屏幕截屏; - 前端增加 tailwind; - 前端 style lint。
This commit is contained in:
parent
fbb222ad23
commit
9814247f4e
7
.eslintignore
Normal file
7
.eslintignore
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
node_modules
|
||||||
|
.DS_Store
|
||||||
|
dist
|
||||||
|
dist-ssr
|
||||||
|
*.local
|
||||||
|
node_modules/*
|
||||||
|
src-tauri
|
52
.eslintrc.cjs
Normal file
52
.eslintrc.cjs
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
module.exports = {
|
||||||
|
root: true,
|
||||||
|
parser: '@typescript-eslint/parser',
|
||||||
|
parserOptions: {
|
||||||
|
ecmaVersion: 2020,
|
||||||
|
sourceType: 'module',
|
||||||
|
ecmaFeatures: {
|
||||||
|
jsx: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
settings: {
|
||||||
|
react: {
|
||||||
|
version: 'detect',
|
||||||
|
},
|
||||||
|
'import/resolver': {
|
||||||
|
node: {
|
||||||
|
paths: ['src'],
|
||||||
|
extensions: ['.js', '.jsx', '.ts', '.tsx'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
env: {
|
||||||
|
browser: true,
|
||||||
|
amd: true,
|
||||||
|
node: true,
|
||||||
|
},
|
||||||
|
extends: [
|
||||||
|
'eslint:recommended',
|
||||||
|
'plugin:@typescript-eslint/recommended',
|
||||||
|
'plugin:react/recommended',
|
||||||
|
'plugin:jsx-a11y/recommended',
|
||||||
|
'plugin:prettier/recommended', // Make sure this is always the last element in the array.
|
||||||
|
],
|
||||||
|
plugins: ['simple-import-sort', 'prettier'],
|
||||||
|
rules: {
|
||||||
|
'prettier/prettier': ['error', {}, { usePrettierrc: true }],
|
||||||
|
'react/react-in-jsx-scope': 'off',
|
||||||
|
'jsx-a11y/accessible-emoji': 'off',
|
||||||
|
'react/prop-types': 'off',
|
||||||
|
'@typescript-eslint/explicit-function-return-type': 'off',
|
||||||
|
'simple-import-sort/imports': 'error',
|
||||||
|
'simple-import-sort/exports': 'error',
|
||||||
|
'jsx-a11y/anchor-is-valid': [
|
||||||
|
'error',
|
||||||
|
{
|
||||||
|
components: ['Link'],
|
||||||
|
specialLink: ['hrefLeft', 'hrefRight'],
|
||||||
|
aspects: ['invalidHref', 'preferButton'],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
};
|
7
.prettierignore
Normal file
7
.prettierignore
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
node_modules
|
||||||
|
.DS_Store
|
||||||
|
dist
|
||||||
|
dist-ssr
|
||||||
|
*.local
|
||||||
|
node_modules/*
|
||||||
|
src-tauri
|
8
.prettierrc.cjs
Normal file
8
.prettierrc.cjs
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
module.exports = {
|
||||||
|
semi: true,
|
||||||
|
trailingComma: "all",
|
||||||
|
singleQuote: true,
|
||||||
|
printWidth: 90,
|
||||||
|
tabWidth: 2,
|
||||||
|
endOfLine: "auto",
|
||||||
|
};
|
17
package.json
17
package.json
@ -10,17 +10,26 @@
|
|||||||
"tauri": "tauri"
|
"tauri": "tauri"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@tauri-apps/api": "^1.1.0",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0"
|
||||||
"@tauri-apps/api": "^1.1.0"
|
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
|
"@tauri-apps/cli": "^1.1.0",
|
||||||
"@types/node": "^18.7.10",
|
"@types/node": "^18.7.10",
|
||||||
"@types/react": "^18.0.15",
|
"@types/react": "^18.0.15",
|
||||||
"@types/react-dom": "^18.0.6",
|
"@types/react-dom": "^18.0.6",
|
||||||
"@vitejs/plugin-react": "^2.0.0",
|
"@vitejs/plugin-react": "^2.0.0",
|
||||||
|
"autoprefixer": "^10.4.13",
|
||||||
|
"eslint-config-prettier": "^8.5.0",
|
||||||
|
"eslint-plugin-import": "^2.26.0",
|
||||||
|
"eslint-plugin-jsx-a11y": "^6.6.1",
|
||||||
|
"eslint-plugin-prettier": "^4.2.1",
|
||||||
|
"eslint-plugin-simple-import-sort": "^8.0.0",
|
||||||
|
"postcss": "^8.4.19",
|
||||||
|
"prettier": "^2.7.1",
|
||||||
|
"tailwindcss": "^3.2.4",
|
||||||
"typescript": "^4.6.4",
|
"typescript": "^4.6.4",
|
||||||
"vite": "^3.0.2",
|
"vite": "^3.0.2"
|
||||||
"@tauri-apps/cli": "^1.1.0"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
1001
pnpm-lock.yaml
1001
pnpm-lock.yaml
File diff suppressed because it is too large
Load Diff
6
postcss.config.cjs
Normal file
6
postcss.config.cjs
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
module.exports = {
|
||||||
|
plugins: {
|
||||||
|
tailwindcss: {},
|
||||||
|
autoprefixer: {},
|
||||||
|
},
|
||||||
|
}
|
9
src-tauri/Cargo.lock
generated
9
src-tauri/Cargo.lock
generated
@ -1742,6 +1742,12 @@ dependencies = [
|
|||||||
"system-deps 6.0.3",
|
"system-deps 6.0.3",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "paris"
|
||||||
|
version = "1.5.13"
|
||||||
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
|
checksum = "2eaf2319cd71dd9ff38c72bebde61b9ea657134abcf26ae4205f54f772a32810"
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "parking_lot"
|
name = "parking_lot"
|
||||||
version = "0.12.1"
|
version = "0.12.1"
|
||||||
@ -2289,8 +2295,11 @@ dependencies = [
|
|||||||
name = "screen-bg-light-desktop"
|
name = "screen-bg-light-desktop"
|
||||||
version = "0.0.0"
|
version = "0.0.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
|
"anyhow",
|
||||||
"base64",
|
"base64",
|
||||||
"bmp",
|
"bmp",
|
||||||
|
"once_cell",
|
||||||
|
"paris",
|
||||||
"repng",
|
"repng",
|
||||||
"scrap",
|
"scrap",
|
||||||
"serde",
|
"serde",
|
||||||
|
@ -22,6 +22,9 @@ repng = "0.2.2"
|
|||||||
bmp = "0.5.0"
|
bmp = "0.5.0"
|
||||||
webp = "0.2.2"
|
webp = "0.2.2"
|
||||||
base64 = "0.13.1"
|
base64 = "0.13.1"
|
||||||
|
anyhow = "1.0.66"
|
||||||
|
once_cell = "1.16.0"
|
||||||
|
paris = { version = "1.5", features = ["timestamps", "macros"] }
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
# by default Tauri runs in production mode
|
# by default Tauri runs in production mode
|
||||||
|
@ -3,13 +3,11 @@
|
|||||||
windows_subsystem = "windows"
|
windows_subsystem = "windows"
|
||||||
)]
|
)]
|
||||||
|
|
||||||
use bmp::{Image, Pixel};
|
mod screen_color_picker;
|
||||||
use scrap::{Capturer, Display};
|
|
||||||
use std::io::ErrorKind::WouldBlock;
|
use paris::*;
|
||||||
use std::path::Path;
|
use screen_color_picker::{Screen, ScreenColorPicker};
|
||||||
use std::thread;
|
use std::time::Instant;
|
||||||
use std::time::Duration;
|
|
||||||
use std::{fs, time::Instant};
|
|
||||||
|
|
||||||
// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command
|
// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
@ -18,65 +16,77 @@ fn greet(name: &str) -> String {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[tauri::command]
|
#[tauri::command]
|
||||||
fn take_snapshot() -> String {
|
fn refresh_displays() {
|
||||||
let one_second = Duration::new(1, 0);
|
match ScreenColorPicker::global().refresh_displays() {
|
||||||
let one_frame = one_second / 60;
|
Ok(_) => {}
|
||||||
|
Err(error) => {
|
||||||
|
error!("{}", error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let display = Display::primary().expect("Couldn't find primary display.");
|
#[tauri::command]
|
||||||
let mut capturer = Capturer::new(display).expect("Couldn't begin capture.");
|
fn take_snapshot() -> Vec<String> {
|
||||||
let (w, h) = (capturer.width(), capturer.height());
|
let start = Instant::now();
|
||||||
|
|
||||||
loop {
|
let screenshots = match ScreenColorPicker::global().take_screenshots_for_all() {
|
||||||
// Wait until there's a frame.
|
Ok(bitmaps) => {
|
||||||
|
info!("bitmaps len: {}", bitmaps.len());
|
||||||
let start = Instant::now();
|
match ScreenColorPicker::global().screens.lock() {
|
||||||
let buffer = match capturer.frame() {
|
Ok(screens) => Vec::from_iter(
|
||||||
Ok(buffer) => buffer,
|
screens
|
||||||
Err(error) => {
|
.iter()
|
||||||
if error.kind() == WouldBlock {
|
.enumerate()
|
||||||
// Keep spinning.
|
.map(|(index, screen)| bitmap_to_webp_base64(screen, &bitmaps[index])),
|
||||||
thread::sleep(one_frame);
|
),
|
||||||
continue;
|
Err(error) => {
|
||||||
} else {
|
error!("can not lock screens. {}", error);
|
||||||
panic!("Error: {}", error);
|
Vec::<String>::new()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
|
||||||
println!("Captured! Saving...");
|
|
||||||
|
|
||||||
// Flip the ARGB image into a BGRA image.
|
|
||||||
|
|
||||||
let mut bitflipped = Vec::with_capacity(w * h * 3);
|
|
||||||
let stride = buffer.len() / h;
|
|
||||||
|
|
||||||
let mut img = Image::new(w as u32, h as u32);
|
|
||||||
|
|
||||||
for y in 0..h {
|
|
||||||
for x in 0..w {
|
|
||||||
let i = stride * y + 4 * x;
|
|
||||||
bitflipped.extend_from_slice(&[buffer[i + 2], buffer[i + 1], buffer[i]]);
|
|
||||||
// img.set_pixel(
|
|
||||||
// x as u32,
|
|
||||||
// y as u32,
|
|
||||||
// Pixel::new(buffer[i + 2], buffer[i + 1], buffer[i]),
|
|
||||||
// );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
Err(error) => {
|
||||||
|
error!("can not take screenshots for all. {}", error);
|
||||||
|
Vec::<String>::new()
|
||||||
|
}
|
||||||
|
};
|
||||||
|
println!("运行耗时: {:?}", start.elapsed());
|
||||||
|
screenshots
|
||||||
|
}
|
||||||
|
|
||||||
let webp_memory =
|
fn bitmap_to_webp_base64(screen: &Screen, bitmap: &Vec<u8>) -> String {
|
||||||
webp::Encoder::from_rgb(bitflipped.as_slice(), w as u32, h as u32).encode(100.0);
|
let mut bitflipped = Vec::with_capacity(screen.width * screen.height * 3);
|
||||||
let encode_text = base64::encode(&*webp_memory);
|
let stride = bitmap.len() / screen.height;
|
||||||
// let output_path = Path::new("assets").join("lake").with_extension("webp");
|
|
||||||
// std::fs::write(&output_path, &*webp_memory).unwrap();
|
// let mut img = Image::new(w as u32, h as u32);
|
||||||
println!("运行耗时: {:?}", start.elapsed());
|
for y in 0..screen.height {
|
||||||
return encode_text;
|
for x in 0..screen.width {
|
||||||
|
let i = stride * y + 4 * x;
|
||||||
|
bitflipped.extend_from_slice(&[bitmap[i + 2], bitmap[i + 1], bitmap[i]]);
|
||||||
|
// img.set_pixel(
|
||||||
|
// x as u32,
|
||||||
|
// y as u32,
|
||||||
|
// Pixel::new(buffer[i + 2], buffer[i + 1], buffer[i]),
|
||||||
|
// );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let webp_memory = webp::Encoder::from_rgb(
|
||||||
|
bitflipped.as_slice(),
|
||||||
|
screen.width as u32,
|
||||||
|
screen.height as u32,
|
||||||
|
)
|
||||||
|
.encode(100.0);
|
||||||
|
return base64::encode(&*webp_memory);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
tauri::Builder::default()
|
tauri::Builder::default()
|
||||||
.invoke_handler(tauri::generate_handler![greet, take_snapshot])
|
.invoke_handler(tauri::generate_handler![
|
||||||
|
greet,
|
||||||
|
take_snapshot,
|
||||||
|
refresh_displays
|
||||||
|
])
|
||||||
.run(tauri::generate_context!())
|
.run(tauri::generate_context!())
|
||||||
.expect("error while running tauri application");
|
.expect("error while running tauri application");
|
||||||
}
|
}
|
||||||
|
96
src-tauri/src/screen_color_picker.rs
Normal file
96
src-tauri/src/screen_color_picker.rs
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
use once_cell::sync::OnceCell;
|
||||||
|
use scrap::{Capturer, Display};
|
||||||
|
use std::{
|
||||||
|
io,
|
||||||
|
sync::{Arc, Mutex},
|
||||||
|
};
|
||||||
|
use paris::*;
|
||||||
|
|
||||||
|
pub struct Screen {
|
||||||
|
bitmap: Option<Vec<u8>>,
|
||||||
|
capturer: Option<Capturer>,
|
||||||
|
init_error: Option<io::Error>,
|
||||||
|
pub height: usize,
|
||||||
|
pub width: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe impl Send for Screen {}
|
||||||
|
|
||||||
|
|
||||||
|
pub struct ScreenColorPicker {
|
||||||
|
pub screens: Arc<Mutex<Vec<Screen>>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ScreenColorPicker {
|
||||||
|
pub fn global() -> &'static ScreenColorPicker {
|
||||||
|
static SCREEN_COLOR_PICKER: OnceCell<ScreenColorPicker> = OnceCell::new();
|
||||||
|
|
||||||
|
SCREEN_COLOR_PICKER.get_or_init(|| ScreenColorPicker {
|
||||||
|
screens: Arc::new(Mutex::new(Vec::<Screen>::new())),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn refresh_displays(&self) -> anyhow::Result<()> {
|
||||||
|
let displays = Display::all()
|
||||||
|
.map_err(|error| anyhow::anyhow!("Can not get all of displays. {}", error))?;
|
||||||
|
let mut screens = self
|
||||||
|
.screens
|
||||||
|
.lock()
|
||||||
|
.map_err(|error| anyhow::anyhow!("lock screens failed. {}", error))?;
|
||||||
|
screens.clear();
|
||||||
|
info!("number of displays: {}", displays.len());
|
||||||
|
for display in displays {
|
||||||
|
let height = display.height();
|
||||||
|
let width = display.width();
|
||||||
|
match Capturer::new(display) {
|
||||||
|
Ok(capturer) => screens.push(Screen {
|
||||||
|
bitmap: None,
|
||||||
|
capturer: Some(capturer),
|
||||||
|
init_error: None,
|
||||||
|
height,
|
||||||
|
width,
|
||||||
|
}),
|
||||||
|
Err(error) => screens.push(Screen {
|
||||||
|
bitmap: None,
|
||||||
|
capturer: None,
|
||||||
|
init_error: Some(error),
|
||||||
|
height,
|
||||||
|
width,
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn take_screenshots_for_all(&self) -> anyhow::Result<Vec<Vec<u8>>> {
|
||||||
|
let mut screens = self
|
||||||
|
.screens
|
||||||
|
.lock()
|
||||||
|
.map_err(|error| anyhow::anyhow!("lock screens failed. {}", error))?;
|
||||||
|
let mut screenshots = Vec::<Vec<u8>>::new();
|
||||||
|
for screen in screens.iter_mut() {
|
||||||
|
let screenshot = screen.take().map_err(|error| {
|
||||||
|
anyhow::anyhow!("take screenshot for display failed. {}", error)
|
||||||
|
})?;
|
||||||
|
screenshots.push(screenshot);
|
||||||
|
}
|
||||||
|
|
||||||
|
anyhow::Ok(screenshots)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Screen {
|
||||||
|
fn take(&mut self) -> anyhow::Result<Vec<u8>> {
|
||||||
|
match self.capturer.as_mut() {
|
||||||
|
Some(capturer) => {
|
||||||
|
let buffer = capturer
|
||||||
|
.frame()
|
||||||
|
.map_err(|error| anyhow::anyhow!("failed to frame of display. {}", error))?;
|
||||||
|
|
||||||
|
self.bitmap = Some(buffer.to_vec());
|
||||||
|
anyhow::Ok(buffer.to_vec())
|
||||||
|
}
|
||||||
|
None => anyhow::bail!("Do not initialized"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
60
src/App.tsx
60
src/App.tsx
@ -1,29 +1,33 @@
|
|||||||
import { useState } from "react";
|
import { useCallback, useState } from 'react';
|
||||||
import reactLogo from "./assets/react.svg";
|
import reactLogo from './assets/react.svg';
|
||||||
import { invoke } from "@tauri-apps/api/tauri";
|
import { invoke } from '@tauri-apps/api/tauri';
|
||||||
import "./App.css";
|
import './App.css';
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const [greetMsg, setGreetMsg] = useState("");
|
const [greetMsg, setGreetMsg] = useState('');
|
||||||
const [name, setName] = useState("");
|
const [name, setName] = useState('');
|
||||||
const [screenshots, setScreenshots] = useState<string[]>([]);
|
const [screenshots, setScreenshots] = useState<string[]>([]);
|
||||||
|
|
||||||
async function greet() {
|
async function greet() {
|
||||||
// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command
|
// Learn more about Tauri commands at https://tauri.app/v1/guides/features/command
|
||||||
setGreetMsg(await invoke("greet", { name }));
|
setGreetMsg(await invoke('greet', { name }));
|
||||||
}
|
}
|
||||||
|
|
||||||
async function takeSnapshot() {
|
async function takeSnapshot() {
|
||||||
const base64Text = await invoke("take_snapshot");
|
const base64TextList: string[] = await invoke('take_snapshot');
|
||||||
|
|
||||||
setScreenshots([`data:image/webp;base64,${base64Text}`]);
|
setScreenshots(base64TextList.map((text) => `data:image/webp;base64,${text}`));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const refreshDisplays = useCallback(async () => {
|
||||||
|
await invoke('refresh_displays');
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="container">
|
<div className="container">
|
||||||
<h1>Welcome to Tauri!</h1>
|
<h1>Welcome to Tauri!</h1>
|
||||||
|
|
||||||
<div className="row">
|
<div className="flex gap-5 justify-center">
|
||||||
<a href="https://vitejs.dev" target="_blank">
|
<a href="https://vitejs.dev" target="_blank">
|
||||||
<img src="/vite.svg" className="logo vite" alt="Vite logo" />
|
<img src="/vite.svg" className="logo vite" alt="Vite logo" />
|
||||||
</a>
|
</a>
|
||||||
@ -35,23 +39,31 @@ function App() {
|
|||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className='row'>
|
<div className="flex gap-1 justify-center w-screen overflow-hidden">
|
||||||
<img src={screenshots[0]} />
|
{screenshots.map((screenshot) => (
|
||||||
|
<div className="flex-auto">
|
||||||
|
<img src={screenshot} />
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<p>Click on the Tauri, Vite, and React logos to learn more.</p>
|
<p>Click on the Tauri, Vite, and React logos to learn more.</p>
|
||||||
|
|
||||||
<div className="row">
|
<div className="flex gap-5 justify-center ">
|
||||||
<div>
|
<button
|
||||||
<input
|
className="bg-black bg-opacity-20"
|
||||||
id="greet-input"
|
type="button"
|
||||||
onChange={(e) => setName(e.currentTarget.value)}
|
onClick={() => refreshDisplays()}
|
||||||
placeholder="Enter a name..."
|
>
|
||||||
/>
|
Refresh Displays
|
||||||
<button type="button" onClick={() => takeSnapshot()}>
|
</button>
|
||||||
Take Snapshot
|
<button
|
||||||
</button>
|
className="bg-black bg-opacity-20"
|
||||||
</div>
|
type="button"
|
||||||
|
onClick={() => takeSnapshot()}
|
||||||
|
>
|
||||||
|
Take Snapshot
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
<p>{greetMsg}</p>
|
<p>{greetMsg}</p>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,3 +1,7 @@
|
|||||||
|
@tailwind base;
|
||||||
|
@tailwind components;
|
||||||
|
@tailwind utilities;
|
||||||
|
|
||||||
:root {
|
:root {
|
||||||
font-family: Inter, Avenir, Helvetica, Arial, sans-serif;
|
font-family: Inter, Avenir, Helvetica, Arial, sans-serif;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
|
12
tailwind.config.cjs
Normal file
12
tailwind.config.cjs
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
/** @type {import('tailwindcss').Config} */
|
||||||
|
module.exports = {
|
||||||
|
content: [
|
||||||
|
"./index.html",
|
||||||
|
"./src/**/*.{js,ts,jsx,tsx}",
|
||||||
|
],
|
||||||
|
theme: {
|
||||||
|
extend: {
|
||||||
|
},
|
||||||
|
},
|
||||||
|
plugins: [],
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user