Add phpinfo
This commit is contained in:
5
api/php/phpinfo.php
Normal file
5
api/php/phpinfo.php
Normal file
@ -0,0 +1,5 @@
|
||||
<?php
|
||||
|
||||
require_once 'cors.php';
|
||||
|
||||
phpinfo();
|
@ -1,6 +1,5 @@
|
||||
const config = {
|
||||
APP_URL: 'http://localhost:3000',
|
||||
// API_URL: 'http://localhost:8080/v1',
|
||||
APP_URL: 'http://192.168.33.27:3000/',
|
||||
};
|
||||
|
||||
export { config }
|
||||
export { config };
|
||||
|
@ -1,10 +1,10 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
<head>
|
||||
<meta charset="UTF-8"/>
|
||||
<link rel="icon" href="/favicon.ico"/>
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||
<title>Tools</title>
|
||||
<meta charset="UTF-8" />
|
||||
<link rel="icon" href="/favicon.ico" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Tools</title>
|
||||
</head>
|
||||
<body>
|
||||
<div id="app"></div>
|
||||
|
@ -1,6 +1,6 @@
|
||||
module.exports = {
|
||||
plugins: {
|
||||
tailwindcss: {},
|
||||
autoprefixer: {},
|
||||
},
|
||||
}
|
||||
plugins: {
|
||||
tailwindcss: {},
|
||||
autoprefixer: {},
|
||||
},
|
||||
};
|
||||
|
125
src/App.vue
125
src/App.vue
@ -1,71 +1,90 @@
|
||||
<template>
|
||||
<div
|
||||
class="w-full flex flex-col sm:flex-row flex-grow"
|
||||
:class="{ 'sidebar-toggle': sidebarToggle }">
|
||||
<aside
|
||||
class="hidden md:block h-screen overflow-y-auto no-scrollbar border-r">
|
||||
<Sidebar/>
|
||||
</aside>
|
||||
<div
|
||||
class="w-full flex flex-col sm:flex-row flex-grow"
|
||||
:class="{ 'sidebar-toggle': sidebarToggle }"
|
||||
>
|
||||
<aside
|
||||
class="hidden md:block h-screen overflow-y-auto no-scrollbar border-r"
|
||||
>
|
||||
<Sidebar />
|
||||
</aside>
|
||||
|
||||
<main role="main" class="w-full h-full flex-grow overflow-auto p-3">
|
||||
<RouterView/>
|
||||
</main>
|
||||
<main role="main" class="w-full h-full flex-grow overflow-auto p-3">
|
||||
<RouterView />
|
||||
</main>
|
||||
|
||||
<button class="sidebar-toggle-btn md:hidden" v-on:click="toggleSidebar">
|
||||
<svg class="svg-icon" style="width: 30px; height: 30px;vertical-align: middle;fill: currentColor;overflow: hidden;" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"><path d="M682.666667 682.666667h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666H170.666667v-170.666666z m512-256h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666H170.666667v-170.666666z m512-256h170.666666v170.666666h-170.666666V170.666667z m-256 0h170.666666v170.666666h-170.666666V170.666667zM170.666667 170.666667h170.666666v170.666666H170.666667V170.666667z" fill="#000000" /></svg>
|
||||
</button>
|
||||
</div>
|
||||
<button class="sidebar-toggle-btn md:hidden" v-on:click="toggleSidebar">
|
||||
<svg
|
||||
class="svg-icon"
|
||||
style="
|
||||
width: 30px;
|
||||
height: 30px;
|
||||
vertical-align: middle;
|
||||
fill: currentColor;
|
||||
overflow: hidden;
|
||||
"
|
||||
viewBox="0 0 1024 1024"
|
||||
version="1.1"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
>
|
||||
<path
|
||||
d="M682.666667 682.666667h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666H170.666667v-170.666666z m512-256h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666h-170.666666v-170.666666z m-256 0h170.666666v170.666666H170.666667v-170.666666z m512-256h170.666666v170.666666h-170.666666V170.666667z m-256 0h170.666666v170.666666h-170.666666V170.666667zM170.666667 170.666667h170.666666v170.666666H170.666667V170.666667z"
|
||||
fill="#000000"
|
||||
/>
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import { RouterView } from 'vue-router'
|
||||
import Sidebar from '@/components/Sidebar.vue';
|
||||
import { RouterView } from "vue-router";
|
||||
import Sidebar from "@/components/Sidebar.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
Sidebar,
|
||||
RouterView,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
sidebarToggle: false,
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
toggleSidebar() {
|
||||
this.sidebarToggle = !this.sidebarToggle;
|
||||
components: {
|
||||
Sidebar,
|
||||
RouterView,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
sidebarToggle: false,
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
toggleSidebar() {
|
||||
this.sidebarToggle = !this.sidebarToggle;
|
||||
|
||||
this.emitter.emit('sidebar.toggle');
|
||||
},
|
||||
},
|
||||
}
|
||||
this.emitter.emit("sidebar.toggle");
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
aside {
|
||||
width: 480px;
|
||||
width: 480px;
|
||||
}
|
||||
@media (max-width: 768px) {
|
||||
.sidebar-toggle {
|
||||
aside {
|
||||
display: block;
|
||||
width: 100%;
|
||||
max-width: initial;
|
||||
}
|
||||
.sidebar-toggle {
|
||||
aside {
|
||||
display: block;
|
||||
width: 100%;
|
||||
max-width: initial;
|
||||
}
|
||||
|
||||
main {
|
||||
display: none;
|
||||
}
|
||||
main {
|
||||
display: none;
|
||||
}
|
||||
|
||||
&-btn {
|
||||
right: 10px;
|
||||
bottom: 10px;
|
||||
position: fixed;
|
||||
z-index: 1000;
|
||||
background-color: white;
|
||||
border-radius: 5px;
|
||||
padding: 2px;
|
||||
}
|
||||
}
|
||||
&-btn {
|
||||
right: 10px;
|
||||
bottom: 10px;
|
||||
position: fixed;
|
||||
z-index: 1000;
|
||||
background-color: white;
|
||||
border-radius: 5px;
|
||||
padding: 2px;
|
||||
}
|
||||
}
|
||||
}
|
||||
</style>
|
||||
</style>
|
||||
|
26
src/app.js
26
src/app.js
@ -1,23 +1,23 @@
|
||||
import { createApp } from 'vue'
|
||||
import { createPinia } from 'pinia'
|
||||
import axios from 'axios'
|
||||
import VueAxios from 'vue-axios'
|
||||
import mitt from 'mitt';
|
||||
import { createApp } from "vue";
|
||||
import { createPinia } from "pinia";
|
||||
import axios from "axios";
|
||||
import VueAxios from "vue-axios";
|
||||
import mitt from "mitt";
|
||||
|
||||
import './assets/app.scss'
|
||||
import "./assets/app.scss";
|
||||
|
||||
import App from './App.vue'
|
||||
import router from './router'
|
||||
import App from "./App.vue";
|
||||
import router from "./router";
|
||||
|
||||
axios.defaults.withCredentials = true;
|
||||
|
||||
const app = createApp(App)
|
||||
const app = createApp(App);
|
||||
|
||||
app.config.globalProperties.emitter = mitt();
|
||||
|
||||
app.use(createPinia())
|
||||
app.use(VueAxios, axios)
|
||||
app.use(createPinia());
|
||||
app.use(VueAxios, axios);
|
||||
|
||||
app.use(router)
|
||||
app.use(router);
|
||||
|
||||
app.mount('#app')
|
||||
app.mount("#app");
|
||||
|
@ -1,9 +1,9 @@
|
||||
<template>
|
||||
<div :id="name"></div>
|
||||
<div :id="name"></div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import loader from '@monaco-editor/loader';
|
||||
import loader from "@monaco-editor/loader";
|
||||
|
||||
let monacoResultInst = null;
|
||||
|
||||
@ -16,11 +16,11 @@ export default {
|
||||
},
|
||||
language: {
|
||||
type: String,
|
||||
default: 'text',
|
||||
default: "text",
|
||||
},
|
||||
value: {
|
||||
type: String,
|
||||
default: '',
|
||||
default: "",
|
||||
},
|
||||
readonly: {
|
||||
type: Boolean,
|
||||
@ -40,7 +40,7 @@ export default {
|
||||
},
|
||||
unmounted() {
|
||||
if (monacoResultInst) {
|
||||
monacoResultInst.editor.getModels().forEach(model => model.dispose());
|
||||
monacoResultInst.editor.getModels().forEach((model) => model.dispose());
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
@ -50,16 +50,14 @@ export default {
|
||||
deep: true,
|
||||
async handler(value, oldValue) {
|
||||
while (!monacoResultInst) {
|
||||
await new Promise(r => setTimeout(r, 10));
|
||||
await new Promise((r) => setTimeout(r, 10));
|
||||
}
|
||||
|
||||
monacoResultInst.editor.getModels()[0].setValue(value.toString());
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
<style scoped></style>
|
||||
|
@ -1,30 +1,53 @@
|
||||
<template>
|
||||
<header class="flex justify-between mx-2 mt-1">
|
||||
<span class="font-bold">Tools</span>
|
||||
<router-link :to="{name:'home'}" class="ml-4" tag="button">Home</router-link>
|
||||
</header>
|
||||
<header class="flex justify-between mx-2 mt-1">
|
||||
<span class="font-bold">Tools</span>
|
||||
<router-link :to="{ name: 'home' }" class="ml-4" tag="button"
|
||||
>Home</router-link
|
||||
>
|
||||
</header>
|
||||
|
||||
<hr class="mt-2 mb-2">
|
||||
<hr class="mt-2 mb-2" />
|
||||
|
||||
<nav class="flex flex-col ml-2">
|
||||
<div v-for="[title, routes] of Object.entries(menuRoutes).sort((a, b) => a[0].localeCompare(b[0]))" class="flex flex-col mb-2">
|
||||
<p><i>{{ title }}</i></p>
|
||||
<router-link v-for="[key, value] of Object.entries(routes)" :to="{name:key}" class="ml-4" tag="button">{{ value }}</router-link>
|
||||
<div
|
||||
v-for="[title, routes] of Object.entries(menuRoutes).sort((a, b) =>
|
||||
a[0].localeCompare(b[0])
|
||||
)"
|
||||
class="flex flex-col mb-2"
|
||||
>
|
||||
<p>
|
||||
<i>{{ title }}</i>
|
||||
</p>
|
||||
<router-link
|
||||
v-for="[key, value] of Object.entries(routes)"
|
||||
:to="{ name: key }"
|
||||
class="ml-4"
|
||||
tag="button"
|
||||
>{{ value }}</router-link
|
||||
>
|
||||
</div>
|
||||
|
||||
<hr class="mt-3 mb-3">
|
||||
<hr class="mt-3 mb-3" />
|
||||
|
||||
<div class="flex flex-col mb-2">
|
||||
<p><i>Other</i></p>
|
||||
<a href="https://gist.stuzer.link/stuzer05/liked" class="ml-4" tag="button">Gist</a>
|
||||
<a href="https://cyberchef.tools.stuzer.link/" class="ml-4" tag="button">Cyberchef</a>
|
||||
<a href="https://pdf.tools.stuzer.link/" class="ml-4" tag="button">PDF tools</a>
|
||||
<a
|
||||
href="https://gist.stuzer.link/stuzer05/liked"
|
||||
class="ml-4"
|
||||
tag="button"
|
||||
>Gist</a
|
||||
>
|
||||
<a href="https://cyberchef.tools.stuzer.link/" class="ml-4" tag="button"
|
||||
>Cyberchef</a
|
||||
>
|
||||
<a href="https://pdf.tools.stuzer.link/" class="ml-4" tag="button"
|
||||
>PDF tools</a
|
||||
>
|
||||
</div>
|
||||
</nav>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
|
||||
import { useToolsStore } from "@/stores/toolsStore";
|
||||
|
||||
export default {
|
||||
@ -37,6 +60,6 @@ export default {
|
||||
},
|
||||
mounted() {
|
||||
this.menuRoutes = useToolsStore().tools;
|
||||
}
|
||||
}
|
||||
</script>
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
@ -9,57 +9,57 @@ const router = createRouter({
|
||||
{
|
||||
path: "/",
|
||||
name: "home",
|
||||
component: () => import("../views/HomeView.vue")
|
||||
component: () => import("../views/HomeView.vue"),
|
||||
},
|
||||
{
|
||||
path: "/explain_crontab",
|
||||
name: "explain_crontab",
|
||||
component: () => import("../views/unix/ExplainCrontab.vue")
|
||||
component: () => import("../views/unix/ExplainCrontab.vue"),
|
||||
},
|
||||
{
|
||||
path: "/table_to_markdown_table",
|
||||
name: "table_to_markdown_table",
|
||||
component: () => import("../views/general/TableToMarkdownTable.vue")
|
||||
component: () => import("../views/general/TableToMarkdownTable.vue"),
|
||||
},
|
||||
{
|
||||
path: "/table_to_mediawiki_table",
|
||||
name: "table_to_mediawiki_table",
|
||||
component: () => import("../views/general/TableToMediawikiTable.vue")
|
||||
component: () => import("../views/general/TableToMediawikiTable.vue"),
|
||||
},
|
||||
{
|
||||
path: "/dummy_image",
|
||||
name: "dummy_image",
|
||||
component: () => import("../views/generators/DummyImage.vue")
|
||||
component: () => import("../views/generators/DummyImage.vue"),
|
||||
},
|
||||
{
|
||||
path: "/humans_txt",
|
||||
name: "humans_txt",
|
||||
component: () => import("../views/general/HumansTxt.vue")
|
||||
component: () => import("../views/general/HumansTxt.vue"),
|
||||
},
|
||||
{
|
||||
path: "/qr_code",
|
||||
name: "qr_code",
|
||||
component: () => import("../views/generators/QRCode.vue")
|
||||
component: () => import("../views/generators/QRCode.vue"),
|
||||
},
|
||||
{
|
||||
path: "/unix_timestamp",
|
||||
name: "unix_timestamp",
|
||||
component: () => import("../views/unix/UnixTimestamp.vue")
|
||||
component: () => import("../views/unix/UnixTimestamp.vue"),
|
||||
},
|
||||
{
|
||||
path: "/file_base64_encode_decode",
|
||||
name: "file_base64_encode_decode",
|
||||
component: () => import("../views/unix/FileBase64EncodeDecode.vue")
|
||||
component: () => import("../views/unix/FileBase64EncodeDecode.vue"),
|
||||
},
|
||||
{
|
||||
path: "/sed_generator",
|
||||
name: "sed_generator",
|
||||
component: () => import("../views/unix/SedGenerator.vue")
|
||||
component: () => import("../views/unix/SedGenerator.vue"),
|
||||
},
|
||||
{
|
||||
path: "/htaccess_generator",
|
||||
name: "htaccess_generator",
|
||||
component: () => import("../views/unix/HtaccessGenerator.vue")
|
||||
component: () => import("../views/unix/HtaccessGenerator.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
@ -68,57 +68,57 @@ const router = createRouter({
|
||||
{
|
||||
path: "/str_length",
|
||||
name: "str_length",
|
||||
component: () => import("../views/strings/Length.vue")
|
||||
component: () => import("../views/strings/Length.vue"),
|
||||
},
|
||||
{
|
||||
path: "/str_sort_lines",
|
||||
name: "str_sort_lines",
|
||||
component: () => import("../views/strings/SortLines.vue")
|
||||
component: () => import("../views/strings/SortLines.vue"),
|
||||
},
|
||||
{
|
||||
path: "/str_to_lower_upper",
|
||||
name: "str_to_lower_upper",
|
||||
component: () => import("../views/strings/ToLowerUppper.vue")
|
||||
component: () => import("../views/strings/ToLowerUppper.vue"),
|
||||
},
|
||||
{
|
||||
path: "/str_remove_duplicate_lines",
|
||||
name: "str_remove_duplicate_lines",
|
||||
component: () => import("../views/strings/RemoveDuplicateLines.vue")
|
||||
component: () => import("../views/strings/RemoveDuplicateLines.vue"),
|
||||
},
|
||||
{
|
||||
path: "/str_pad",
|
||||
name: "str_pad",
|
||||
component: () => import("../views/strings/Pad.vue")
|
||||
component: () => import("../views/strings/Pad.vue"),
|
||||
},
|
||||
{
|
||||
path: "/str_numeronym",
|
||||
name: "str_numeronym",
|
||||
component: () => import("../views/strings/Numeronym.vue")
|
||||
component: () => import("../views/strings/Numeronym.vue"),
|
||||
},
|
||||
{
|
||||
path: "/str_to_nato_alphabet",
|
||||
name: "str_to_nato_alphabet",
|
||||
component: () => import("../views/strings/NATOAlphabet.vue")
|
||||
component: () => import("../views/strings/NATOAlphabet.vue"),
|
||||
},
|
||||
{
|
||||
path: "/url_encode_decode",
|
||||
name: "url_encode_decode",
|
||||
component: () => import("../views/strings/UrlEncodeDecode.vue")
|
||||
component: () => import("../views/strings/UrlEncodeDecode.vue"),
|
||||
},
|
||||
{
|
||||
path: "/url_query_viewer",
|
||||
name: "url_query_viewer",
|
||||
component: () => import("../views/strings/UrlQueryViewer.vue")
|
||||
component: () => import("../views/strings/UrlQueryViewer.vue"),
|
||||
},
|
||||
{
|
||||
path: "/fix_ru_en_keyboard",
|
||||
name: "fix_ru_en_keyboard",
|
||||
component: () => import("../views/strings/FixRuEnKeyboard.vue")
|
||||
component: () => import("../views/strings/FixRuEnKeyboard.vue"),
|
||||
},
|
||||
{
|
||||
path: "/iban_generator",
|
||||
name: "iban_generator",
|
||||
component: () => import("../views/generators/IbanGenerator.vue")
|
||||
component: () => import("../views/generators/IbanGenerator.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
@ -127,17 +127,17 @@ const router = createRouter({
|
||||
{
|
||||
path: "/str_to_php_array",
|
||||
name: "str_to_php_array",
|
||||
component: () => import("../views/php/StrToPHPArray.vue")
|
||||
component: () => import("../views/php/StrToPHPArray.vue"),
|
||||
},
|
||||
{
|
||||
path: "/php_array_to_json",
|
||||
name: "php_array_to_json",
|
||||
component: () => import("../views/php/PHPArrayToJson.vue")
|
||||
component: () => import("../views/php/PHPArrayToJson.vue"),
|
||||
},
|
||||
{
|
||||
path: "/php_serialize",
|
||||
name: "php_serialize",
|
||||
component: () => import("../views/php/Serialize.vue")
|
||||
component: () => import("../views/php/Serialize.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
@ -146,12 +146,12 @@ const router = createRouter({
|
||||
{
|
||||
path: "/json_formatter",
|
||||
name: "json_formatter",
|
||||
component: () => import("../views/json/JSONFormatter.vue")
|
||||
component: () => import("../views/json/JSONFormatter.vue"),
|
||||
},
|
||||
{
|
||||
path: "/json_minifier",
|
||||
name: "json_minifier",
|
||||
component: () => import("../views/json/JSONMinifier.vue")
|
||||
component: () => import("../views/json/JSONMinifier.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
@ -160,12 +160,12 @@ const router = createRouter({
|
||||
{
|
||||
path: "/go_json_to_struct",
|
||||
name: "go_json_to_struct",
|
||||
component: () => import("../views/go/JSONToStruct.vue")
|
||||
component: () => import("../views/go/JSONToStruct.vue"),
|
||||
},
|
||||
{
|
||||
path: "/sql_tables_to_struct",
|
||||
name: "sql_tables_to_struct",
|
||||
component: () => import("../views/go/SQLTablesToStruct.vue")
|
||||
component: () => import("../views/go/SQLTablesToStruct.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
@ -174,7 +174,7 @@ const router = createRouter({
|
||||
{
|
||||
path: "/docker_rename_volume",
|
||||
name: "docker_rename_volume",
|
||||
component: () => import("../views/docker/RenameVolume.vue")
|
||||
component: () => import("../views/docker/RenameVolume.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
@ -183,14 +183,23 @@ const router = createRouter({
|
||||
{
|
||||
path: "/sql_split_in",
|
||||
name: "sql_split_in",
|
||||
component: () => import("../views/sql/SplitInView.vue")
|
||||
component: () => import("../views/sql/SplitInView.vue"),
|
||||
},
|
||||
{
|
||||
path: "/sql_formatter",
|
||||
name: "sql_formatter",
|
||||
component: () => import("../views/sql/Formatter.vue")
|
||||
}
|
||||
]
|
||||
component: () => import("../views/sql/Formatter.vue"),
|
||||
},
|
||||
|
||||
/**
|
||||
* Other
|
||||
*/
|
||||
{
|
||||
path: "/phpinfo",
|
||||
name: "phpinfo",
|
||||
component: () => import("../views/PHPInfo.vue"),
|
||||
},
|
||||
],
|
||||
});
|
||||
|
||||
export default router;
|
||||
|
@ -1,4 +1,4 @@
|
||||
import { defineStore } from 'pinia'
|
||||
import { defineStore } from "pinia";
|
||||
|
||||
function sortMenuRoutes(routes) {
|
||||
const sortedRoutes = {};
|
||||
@ -17,61 +17,61 @@ function sortMenuRoutes(routes) {
|
||||
return sortedRoutes;
|
||||
}
|
||||
|
||||
export const useToolsStore = defineStore('tools', {
|
||||
export const useToolsStore = defineStore("tools", {
|
||||
state: () => ({
|
||||
tools: sortMenuRoutes({
|
||||
'-': {
|
||||
'table_to_markdown_table': 'Table to Markdown table',
|
||||
'table_to_mediawiki_table': 'Table to Mediawiki table',
|
||||
'humans_txt': 'humans.txt generator',
|
||||
"-": {
|
||||
table_to_markdown_table: "Table to Markdown table",
|
||||
table_to_mediawiki_table: "Table to Mediawiki table",
|
||||
humans_txt: "humans.txt generator",
|
||||
},
|
||||
'Docker': {
|
||||
'docker_rename_volume': 'Rename volume',
|
||||
Docker: {
|
||||
docker_rename_volume: "Rename volume",
|
||||
},
|
||||
'GO': {
|
||||
'go_json_to_struct': 'JSON to Go struct',
|
||||
GO: {
|
||||
go_json_to_struct: "JSON to Go struct",
|
||||
// 'sql_tables_to_struct': 'SQL tables Go struct',
|
||||
},
|
||||
'JSON': {
|
||||
'json_minifier': 'JSON minifier',
|
||||
'json_formatter': 'JSON formatter',
|
||||
JSON: {
|
||||
json_minifier: "JSON minifier",
|
||||
json_formatter: "JSON formatter",
|
||||
},
|
||||
'PHP': {
|
||||
'str_to_php_array': 'Str to PHP array',
|
||||
'php_array_to_json': 'PHP array to Json',
|
||||
'php_serialize': 'PHP serialize',
|
||||
PHP: {
|
||||
str_to_php_array: "Str to PHP array",
|
||||
php_array_to_json: "PHP array to Json",
|
||||
php_serialize: "PHP serialize",
|
||||
},
|
||||
'SQL': {
|
||||
'sql_formatter': 'SQL formatter',
|
||||
'sql_split_in': 'SQL split IN',
|
||||
SQL: {
|
||||
sql_formatter: "SQL formatter",
|
||||
sql_split_in: "SQL split IN",
|
||||
},
|
||||
'Strings': {
|
||||
'fix_ru_en_keyboard': 'Fix ru-en keyboard',
|
||||
'str_length': 'Str length',
|
||||
'str_sort_lines': 'Str sort lines',
|
||||
'str_to_lower_upper': 'Str to lower/upper',
|
||||
'str_remove_duplicate_lines': 'Str remove duplicate lines',
|
||||
'str_pad': 'Str pad',
|
||||
'str_numeronym': 'Str numeronym (i18n)',
|
||||
'str_to_nato_alphabet': 'Str to NATO alphabet',
|
||||
'url_encode_decode': 'URL encode/decode',
|
||||
'url_query_viewer': 'URL query viewer',
|
||||
Strings: {
|
||||
fix_ru_en_keyboard: "Fix ru-en keyboard",
|
||||
str_length: "Str length",
|
||||
str_sort_lines: "Str sort lines",
|
||||
str_to_lower_upper: "Str to lower/upper",
|
||||
str_remove_duplicate_lines: "Str remove duplicate lines",
|
||||
str_pad: "Str pad",
|
||||
str_numeronym: "Str numeronym (i18n)",
|
||||
str_to_nato_alphabet: "Str to NATO alphabet",
|
||||
url_encode_decode: "URL encode/decode",
|
||||
url_query_viewer: "URL query viewer",
|
||||
},
|
||||
'Unix': {
|
||||
'explain_crontab': 'Explain crontab',
|
||||
'file_base64_encode_decode': 'File base64 encode/decode',
|
||||
'unix_timestamp': 'Unix timestamp',
|
||||
'sed_generator': 'Sed generator',
|
||||
'htaccess_generator': '.htaccess generator',
|
||||
Unix: {
|
||||
explain_crontab: "Explain crontab",
|
||||
file_base64_encode_decode: "File base64 encode/decode",
|
||||
unix_timestamp: "Unix timestamp",
|
||||
sed_generator: "Sed generator",
|
||||
htaccess_generator: ".htaccess generator",
|
||||
},
|
||||
'Generators': {
|
||||
'qr_code': 'QR code',
|
||||
'iban_generator': 'IBAN generator',
|
||||
'dummy_image': 'Dummy image',
|
||||
Generators: {
|
||||
qr_code: "QR code",
|
||||
iban_generator: "IBAN generator",
|
||||
dummy_image: "Dummy image",
|
||||
},
|
||||
}),
|
||||
actions: {
|
||||
// You can add actions here if needed, e.g., to update the tools
|
||||
},
|
||||
}),
|
||||
})
|
||||
});
|
||||
|
@ -1,7 +1,5 @@
|
||||
function unproxy(obj) {
|
||||
return JSON.parse(JSON.stringify(obj));
|
||||
return JSON.parse(JSON.stringify(obj));
|
||||
}
|
||||
|
||||
export {
|
||||
unproxy,
|
||||
}
|
||||
export { unproxy };
|
||||
|
@ -1,36 +1,34 @@
|
||||
import { ref, customRef } from 'vue'
|
||||
import { ref, customRef } from "vue";
|
||||
|
||||
const debounce = (fn, delay = 0, immediate = false) => {
|
||||
let timeout
|
||||
return (...args) => {
|
||||
if (immediate && !timeout) fn(...args)
|
||||
clearTimeout(timeout)
|
||||
let timeout;
|
||||
return (...args) => {
|
||||
if (immediate && !timeout) fn(...args);
|
||||
clearTimeout(timeout);
|
||||
|
||||
timeout = setTimeout(() => {
|
||||
fn(...args)
|
||||
}, delay)
|
||||
}
|
||||
}
|
||||
timeout = setTimeout(() => {
|
||||
fn(...args);
|
||||
}, delay);
|
||||
};
|
||||
};
|
||||
|
||||
const useDebouncedRef = (initialValue, delay, immediate) => {
|
||||
const state = ref(initialValue)
|
||||
const debouncedRef = customRef((track, trigger) => ({
|
||||
get() {
|
||||
track()
|
||||
return state.value
|
||||
},
|
||||
set: debounce(
|
||||
value => {
|
||||
state.value = value
|
||||
trigger()
|
||||
},
|
||||
delay,
|
||||
immediate
|
||||
),
|
||||
}))
|
||||
return debouncedRef
|
||||
}
|
||||
const state = ref(initialValue);
|
||||
const debouncedRef = customRef((track, trigger) => ({
|
||||
get() {
|
||||
track();
|
||||
return state.value;
|
||||
},
|
||||
set: debounce(
|
||||
(value) => {
|
||||
state.value = value;
|
||||
trigger();
|
||||
},
|
||||
delay,
|
||||
immediate
|
||||
),
|
||||
}));
|
||||
return debouncedRef;
|
||||
};
|
||||
|
||||
export {
|
||||
useDebouncedRef,
|
||||
}
|
||||
export { useDebouncedRef };
|
||||
|
@ -1,7 +1,13 @@
|
||||
<template>
|
||||
<div class="container mx-auto p-4">
|
||||
<div class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-5 gap-5">
|
||||
<div v-for="(category, categoryName) in tools" :key="categoryName" class="bg-white rounded-lg shadow-md p-4">
|
||||
<div
|
||||
class="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-5 gap-5"
|
||||
>
|
||||
<div
|
||||
v-for="(category, categoryName) in tools"
|
||||
:key="categoryName"
|
||||
class="bg-white rounded-lg shadow-md p-4"
|
||||
>
|
||||
<h2 class="text-xl font-semibold mb-3">{{ categoryName }}</h2>
|
||||
<div class="max-h-80 overflow-y-auto">
|
||||
<ul class="space-y-2">
|
||||
@ -19,22 +25,43 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="my-5">
|
||||
<hr class="my-5" />
|
||||
|
||||
<div class="flex flex-col items-center justify-center p-4">
|
||||
<div class="flex flex-wrap justify-center items-center gap-4 max-w-4xl">
|
||||
<a href="https://gist.stuzer.link/stuzer05/liked" class="flex items-center w-64 p-4 bg-white rounded-lg shadow-md hover:bg-gray-50 transition-colors duration-300">
|
||||
<img src="https://gist.stuzer.link/assets/opengist-85b89b9c.svg" alt="Gist Icon" class="w-12 h-12 mr-4">
|
||||
<a
|
||||
href="https://gist.stuzer.link/stuzer05/liked"
|
||||
class="flex items-center w-64 p-4 bg-white rounded-lg shadow-md hover:bg-gray-50 transition-colors duration-300"
|
||||
>
|
||||
<img
|
||||
src="https://gist.stuzer.link/assets/opengist-85b89b9c.svg"
|
||||
alt="Gist Icon"
|
||||
class="w-12 h-12 mr-4"
|
||||
/>
|
||||
<span class="text-lg font-medium">Gist</span>
|
||||
</a>
|
||||
|
||||
<a href="https://cyberchef.tools.stuzer.link/" class="flex items-center w-64 p-4 bg-white rounded-lg shadow-md hover:bg-gray-50 transition-colors duration-300">
|
||||
<img src="https://cyberchef.tools.stuzer.link/images/cyberchef-128x128.png" alt="Cyberchef Icon" class="w-12 h-12 mr-4">
|
||||
<a
|
||||
href="https://cyberchef.tools.stuzer.link/"
|
||||
class="flex items-center w-64 p-4 bg-white rounded-lg shadow-md hover:bg-gray-50 transition-colors duration-300"
|
||||
>
|
||||
<img
|
||||
src="https://cyberchef.tools.stuzer.link/images/cyberchef-128x128.png"
|
||||
alt="Cyberchef Icon"
|
||||
class="w-12 h-12 mr-4"
|
||||
/>
|
||||
<span class="text-lg font-medium">Cyberchef</span>
|
||||
</a>
|
||||
|
||||
<a href="https://pdf.tools.stuzer.link/" class="flex items-center w-64 p-4 bg-white rounded-lg shadow-md hover:bg-gray-50 transition-colors duration-300">
|
||||
<img src="https://pdf.tools.stuzer.link/favicon.svg" alt="PDF Tools Icon" class="w-12 h-12 mr-4">
|
||||
<a
|
||||
href="https://pdf.tools.stuzer.link/"
|
||||
class="flex items-center w-64 p-4 bg-white rounded-lg shadow-md hover:bg-gray-50 transition-colors duration-300"
|
||||
>
|
||||
<img
|
||||
src="https://pdf.tools.stuzer.link/favicon.svg"
|
||||
alt="PDF Tools Icon"
|
||||
class="w-12 h-12 mr-4"
|
||||
/>
|
||||
<span class="text-lg font-medium">PDF tools</span>
|
||||
</a>
|
||||
</div>
|
||||
@ -49,7 +76,7 @@ export default {
|
||||
data() {
|
||||
return {
|
||||
tools: {},
|
||||
}
|
||||
};
|
||||
},
|
||||
setup() {
|
||||
const router = useRouter();
|
||||
@ -62,10 +89,8 @@ export default {
|
||||
},
|
||||
mounted() {
|
||||
this.tools = useToolsStore().tools;
|
||||
}
|
||||
}
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
55
src/views/PHPInfo.vue
Normal file
55
src/views/PHPInfo.vue
Normal file
@ -0,0 +1,55 @@
|
||||
<template>
|
||||
<div v-html="toolResult"></div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import axios from "axios";
|
||||
import { config } from "../../config";
|
||||
|
||||
export default {
|
||||
data() {
|
||||
return {
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
axios.post(`${config.APP_URL}/api/php/phpinfo.php`, {}).then((response) => {
|
||||
const phpinfoHTML = response.data;
|
||||
|
||||
// Create a temporary DOM element
|
||||
const tempDiv = document.createElement("div");
|
||||
tempDiv.innerHTML = phpinfoHTML;
|
||||
|
||||
// Select all <style> tags
|
||||
const styleTags = tempDiv.querySelectorAll("style");
|
||||
|
||||
styleTags.forEach((styleTag) => {
|
||||
// Get the CSS content
|
||||
let cssContent = styleTag.innerHTML;
|
||||
|
||||
// Prepend "main " to each selector
|
||||
cssContent = cssContent.replace(/([^{}]+){/g, "main $1 {");
|
||||
|
||||
// Update the <style> tag's content
|
||||
styleTag.innerHTML = cssContent;
|
||||
});
|
||||
|
||||
// Make <hr> full width
|
||||
tempDiv.innerHTML += `
|
||||
<style>
|
||||
main hr {
|
||||
width: 100% !important;
|
||||
}
|
||||
</style>
|
||||
`;
|
||||
|
||||
// Extract the modified HTML content
|
||||
this.toolResult = tempDiv.innerHTML;
|
||||
|
||||
Promise.resolve(response);
|
||||
});
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss"></style>
|
@ -1,22 +1,36 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Docker rename volume</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="volume_name_old">Old volume name</label>
|
||||
<input id="volume_name_old" v-model="toolData.volumeNameOld" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="volume_name_old"
|
||||
v-model="toolData.volumeNameOld"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="volume_name_new">New volume name</label>
|
||||
<input id="volume_name_new" v-model="toolData.volumeNameNew" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="volume_name_new"
|
||||
v-model="toolData.volumeNameNew"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -25,15 +39,15 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
volumeNameOld: "",
|
||||
volumeNameNew: ""
|
||||
volumeNameNew: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -48,11 +62,9 @@ export default {
|
||||
}
|
||||
|
||||
this.toolResult = `docker run --rm -v ${this.toolData.volumeNameOld}:/from -v ${this.toolData.volumeNameNew}:/to busybox cp -av /from/. /to`;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,78 +1,174 @@
|
||||
<template>
|
||||
<h2 class="tool-title">humans.txt generator</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label>Team
|
||||
<label
|
||||
>Team
|
||||
<button @click="addTeamGroup">+</button>
|
||||
</label>
|
||||
<div class="humans-group" v-for="(group, groupIndex) in toolData.team" :key="groupIndex">
|
||||
<div
|
||||
class="humans-group"
|
||||
v-for="(group, groupIndex) in toolData.team"
|
||||
:key="groupIndex"
|
||||
>
|
||||
<div v-for="(field, fieldIndex) in group" :key="fieldIndex">
|
||||
<input class="input" v-model="field.key" v-on:keyup="result" placeholder="Key" type="text">
|
||||
<input class="input" v-model="field.value" v-on:keyup="result" placeholder="Value" type="text">
|
||||
<input
|
||||
class="input"
|
||||
v-model="field.key"
|
||||
v-on:keyup="result"
|
||||
placeholder="Key"
|
||||
type="text"
|
||||
/>
|
||||
<input
|
||||
class="input"
|
||||
v-model="field.value"
|
||||
v-on:keyup="result"
|
||||
placeholder="Value"
|
||||
type="text"
|
||||
/>
|
||||
<button @click="removeTeamField(groupIndex, fieldIndex)">🞩</button>
|
||||
<button @click="moveTeamFieldUp(groupIndex, fieldIndex)" v-show="fieldIndex !== 0">↑</button>
|
||||
<button @click="moveTeamFieldDown(groupIndex, fieldIndex)" v-show="fieldIndex !== group.length - 1">↓</button>
|
||||
<button
|
||||
@click="moveTeamFieldUp(groupIndex, fieldIndex)"
|
||||
v-show="fieldIndex !== 0"
|
||||
>
|
||||
↑
|
||||
</button>
|
||||
<button
|
||||
@click="moveTeamFieldDown(groupIndex, fieldIndex)"
|
||||
v-show="fieldIndex !== group.length - 1"
|
||||
>
|
||||
↓
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<button @click="addTeamField(groupIndex)">+</button>
|
||||
<button @click="removeTeamGroup(groupIndex)">🞩</button>
|
||||
<button @click="moveTeamGroupUp(groupIndex)" v-show="groupIndex !== 0">↑</button>
|
||||
<button @click="moveTeamGroupDown(groupIndex)" v-show="groupIndex !== toolData.team.length - 1">↓</button>
|
||||
<button @click="moveTeamGroupUp(groupIndex)" v-show="groupIndex !== 0">
|
||||
↑
|
||||
</button>
|
||||
<button
|
||||
@click="moveTeamGroupDown(groupIndex)"
|
||||
v-show="groupIndex !== toolData.team.length - 1"
|
||||
>
|
||||
↓
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label>Thanks
|
||||
<label
|
||||
>Thanks
|
||||
<button @click="addThanksGroup">+</button>
|
||||
</label>
|
||||
<div class="humans-group" v-for="(group, groupIndex) in toolData.thanks" :key="groupIndex">
|
||||
<div
|
||||
class="humans-group"
|
||||
v-for="(group, groupIndex) in toolData.thanks"
|
||||
:key="groupIndex"
|
||||
>
|
||||
<div v-for="(field, fieldIndex) in group" :key="fieldIndex">
|
||||
<input class="input" v-model="field.key" v-on:keyup="result" placeholder="Key" type="text">
|
||||
<input class="input" v-model="field.value" v-on:keyup="result" placeholder="Value" type="text">
|
||||
<input
|
||||
class="input"
|
||||
v-model="field.key"
|
||||
v-on:keyup="result"
|
||||
placeholder="Key"
|
||||
type="text"
|
||||
/>
|
||||
<input
|
||||
class="input"
|
||||
v-model="field.value"
|
||||
v-on:keyup="result"
|
||||
placeholder="Value"
|
||||
type="text"
|
||||
/>
|
||||
<button @click="removeThanksField(groupIndex, fieldIndex)">🞩</button>
|
||||
<button @click="moveThanksFieldUp(groupIndex, fieldIndex)" v-show="fieldIndex !== 0">↑</button>
|
||||
<button @click="moveThanksFieldDown(groupIndex, fieldIndex)" v-show="fieldIndex !== group.length - 1">↓</button>
|
||||
<button
|
||||
@click="moveThanksFieldUp(groupIndex, fieldIndex)"
|
||||
v-show="fieldIndex !== 0"
|
||||
>
|
||||
↑
|
||||
</button>
|
||||
<button
|
||||
@click="moveThanksFieldDown(groupIndex, fieldIndex)"
|
||||
v-show="fieldIndex !== group.length - 1"
|
||||
>
|
||||
↓
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<button @click="addThanksField(groupIndex)">+</button>
|
||||
<button @click="removeThanksGroup(groupIndex)">🞩</button>
|
||||
<button @click="moveThanksGroupUp(groupIndex)" v-show="groupIndex !== 0">↑</button>
|
||||
<button @click="moveThanksGroupDown(groupIndex)" v-show="groupIndex !== toolData.thanks.length - 1">↓</button>
|
||||
<button @click="moveThanksGroupUp(groupIndex)" v-show="groupIndex !== 0">
|
||||
↑
|
||||
</button>
|
||||
<button
|
||||
@click="moveThanksGroupDown(groupIndex)"
|
||||
v-show="groupIndex !== toolData.thanks.length - 1"
|
||||
>
|
||||
↓
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label>Site info
|
||||
<label
|
||||
>Site info
|
||||
<button @click="addSite">+</button>
|
||||
</label>
|
||||
<div v-for="(info, fieldIndex) in toolData.site" :key="fieldIndex">
|
||||
<input class="input" v-model="info.key" v-on:keyup="result" placeholder="Key" type="text">
|
||||
<input class="input" v-model="info.value" v-on:keyup="result" placeholder="Value" type="text">
|
||||
<input
|
||||
class="input"
|
||||
v-model="info.key"
|
||||
v-on:keyup="result"
|
||||
placeholder="Key"
|
||||
type="text"
|
||||
/>
|
||||
<input
|
||||
class="input"
|
||||
v-model="info.value"
|
||||
v-on:keyup="result"
|
||||
placeholder="Value"
|
||||
type="text"
|
||||
/>
|
||||
<button @click="removeSiteField(fieldIndex)">🞩</button>
|
||||
<button @click="moveSiteUp(fieldIndex)" v-show="fieldIndex !== 0">↑</button>
|
||||
<button @click="moveSiteDown(fieldIndex)" v-show="fieldIndex !== toolData.site.length - 1">↓</button>
|
||||
<button @click="moveSiteUp(fieldIndex)" v-show="fieldIndex !== 0">
|
||||
↑
|
||||
</button>
|
||||
<button
|
||||
@click="moveSiteDown(fieldIndex)"
|
||||
v-show="fieldIndex !== toolData.site.length - 1"
|
||||
>
|
||||
↓
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label>Import humans.txt
|
||||
<label
|
||||
>Import humans.txt
|
||||
<button @click="importHumans">Import</button>
|
||||
</label>
|
||||
<textarea v-model="toolData.importedHumansTxt" style="height: 250px"></textarea>
|
||||
<textarea
|
||||
v-model="toolData.importedHumansTxt"
|
||||
style="height: 250px"
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -81,27 +177,17 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
team: [
|
||||
[
|
||||
{ key: "", value: "" }
|
||||
]
|
||||
],
|
||||
thanks: [
|
||||
[
|
||||
{ key: "", value: "" }
|
||||
]
|
||||
],
|
||||
site: [
|
||||
{ key: "", value: "" }
|
||||
],
|
||||
importedHumansTxt: ""
|
||||
team: [[{ key: "", value: "" }]],
|
||||
thanks: [[{ key: "", value: "" }]],
|
||||
site: [{ key: "", value: "" }],
|
||||
importedHumansTxt: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
@ -152,7 +238,8 @@ export default {
|
||||
moveTeamFieldUp(groupIndex, fieldIndex) {
|
||||
if (fieldIndex > 0) {
|
||||
const temp = this.toolData.team[groupIndex][fieldIndex];
|
||||
this.toolData.team[groupIndex][fieldIndex] = this.toolData.team[groupIndex][fieldIndex - 1];
|
||||
this.toolData.team[groupIndex][fieldIndex] =
|
||||
this.toolData.team[groupIndex][fieldIndex - 1];
|
||||
this.toolData.team[groupIndex][fieldIndex - 1] = temp;
|
||||
}
|
||||
|
||||
@ -161,7 +248,8 @@ export default {
|
||||
moveTeamFieldDown(groupIndex, fieldIndex) {
|
||||
if (fieldIndex < this.toolData.team[groupIndex].length - 1) {
|
||||
const temp = this.toolData.team[groupIndex][fieldIndex];
|
||||
this.toolData.team[groupIndex][fieldIndex] = this.toolData.team[groupIndex][fieldIndex + 1];
|
||||
this.toolData.team[groupIndex][fieldIndex] =
|
||||
this.toolData.team[groupIndex][fieldIndex + 1];
|
||||
this.toolData.team[groupIndex][fieldIndex + 1] = temp;
|
||||
}
|
||||
|
||||
@ -211,7 +299,8 @@ export default {
|
||||
moveThanksFieldUp(groupIndex, fieldIndex) {
|
||||
if (fieldIndex > 0) {
|
||||
const temp = this.toolData.thanks[groupIndex][fieldIndex];
|
||||
this.toolData.thanks[groupIndex][fieldIndex] = this.toolData.thanks[groupIndex][fieldIndex - 1];
|
||||
this.toolData.thanks[groupIndex][fieldIndex] =
|
||||
this.toolData.thanks[groupIndex][fieldIndex - 1];
|
||||
this.toolData.thanks[groupIndex][fieldIndex - 1] = temp;
|
||||
}
|
||||
|
||||
@ -220,7 +309,8 @@ export default {
|
||||
moveThanksFieldDown(groupIndex, fieldIndex) {
|
||||
if (fieldIndex < this.toolData.thanks[groupIndex].length - 1) {
|
||||
const temp = this.toolData.thanks[groupIndex][fieldIndex];
|
||||
this.toolData.thanks[groupIndex][fieldIndex] = this.toolData.thanks[groupIndex][fieldIndex + 1];
|
||||
this.toolData.thanks[groupIndex][fieldIndex] =
|
||||
this.toolData.thanks[groupIndex][fieldIndex + 1];
|
||||
this.toolData.thanks[groupIndex][fieldIndex + 1] = temp;
|
||||
}
|
||||
|
||||
@ -272,7 +362,9 @@ export default {
|
||||
for (const line of lines) {
|
||||
if (line.toLowerCase().startsWith("/* TEAM */".toLowerCase())) {
|
||||
currentSection = "team";
|
||||
} else if (line.toLowerCase().startsWith("/* THANKS */".toLowerCase())) {
|
||||
} else if (
|
||||
line.toLowerCase().startsWith("/* THANKS */".toLowerCase())
|
||||
) {
|
||||
currentSection = "thanks";
|
||||
} else if (line.toLowerCase().startsWith("/* SITE */".toLowerCase())) {
|
||||
currentSection = "site";
|
||||
@ -353,8 +445,8 @@ export default {
|
||||
}
|
||||
|
||||
this.toolResult = output;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
@ -366,6 +458,4 @@ export default {
|
||||
.humans-group:nth-child(odd) {
|
||||
background-color: #f7f7f7;
|
||||
}
|
||||
|
||||
|
||||
</style>
|
||||
</style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Table to Markdown table</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data (paste from excel)</label>
|
||||
@ -20,16 +20,26 @@
|
||||
<label>Transpose table</label>
|
||||
|
||||
<div>
|
||||
<input id="transpose" name="transpose" v-model="toolData.transpose" v-on:change="result" type="checkbox"> <label
|
||||
for="transpose">transpose</label>
|
||||
<input
|
||||
id="transpose"
|
||||
name="transpose"
|
||||
v-model="toolData.transpose"
|
||||
v-on:change="result"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="transpose">transpose</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -39,16 +49,16 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
selectedStyle: "",
|
||||
transpose: false
|
||||
transpose: false,
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -92,11 +102,9 @@ export default {
|
||||
}
|
||||
|
||||
this.toolResult = output;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Table to Mediawiki table</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data (paste from excel)</label>
|
||||
<textarea id="data" v-model="toolData.data" v-on:keyup="result"></textarea>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -21,19 +25,19 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: ""
|
||||
data: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
let output = "{| class=\"wikitable\"\n";
|
||||
let output = '{| class="wikitable"\n';
|
||||
|
||||
const rows = unproxy(this.toolData.data).split("\n");
|
||||
for (let i = 0; i < rows.length; i++) {
|
||||
@ -49,11 +53,9 @@ export default {
|
||||
output += "|}";
|
||||
|
||||
this.toolResult = output;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,35 +1,63 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Dummy image</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label>Dimensions</label>
|
||||
<div style="display: flex">
|
||||
<input class="input" v-model="toolData.width" v-on:keyup="result" v-on:change="result" placeholder="500"
|
||||
type="number" style="width: 70px">px
|
||||
<input
|
||||
class="input"
|
||||
v-model="toolData.width"
|
||||
v-on:keyup="result"
|
||||
v-on:change="result"
|
||||
placeholder="500"
|
||||
type="number"
|
||||
style="width: 70px"
|
||||
/>px
|
||||
<span style="padding: 0 10px">x</span>
|
||||
<input class="input" v-model="toolData.height" v-on:keyup="result" v-on:change="result" placeholder="500"
|
||||
type="number" style="width: 70px">px
|
||||
<input
|
||||
class="input"
|
||||
v-model="toolData.height"
|
||||
v-on:keyup="result"
|
||||
v-on:change="result"
|
||||
placeholder="500"
|
||||
type="number"
|
||||
style="width: 70px"
|
||||
/>px
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Background colour</label>
|
||||
<input class="input" v-model="toolData.color_bg" v-on:change="result" type="color">
|
||||
<input
|
||||
class="input"
|
||||
v-model="toolData.color_bg"
|
||||
v-on:change="result"
|
||||
type="color"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Text colour</label>
|
||||
<input class="input" v-model="toolData.color_text" v-on:change="result" type="color">
|
||||
<input
|
||||
class="input"
|
||||
v-model="toolData.color_text"
|
||||
v-on:change="result"
|
||||
type="color"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<div :style="{'width': toolData.width, 'height': toolData.height}">
|
||||
<canvas ref="canvas" :width="toolData.width" :height="toolData.height"
|
||||
style="max-width: 50vw; max-height: 50vh;"></canvas>
|
||||
<div :style="{ width: toolData.width, height: toolData.height }">
|
||||
<canvas
|
||||
ref="canvas"
|
||||
:width="toolData.width"
|
||||
:height="toolData.height"
|
||||
style="max-width: 50vw; max-height: 50vh"
|
||||
></canvas>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@ -42,9 +70,9 @@ export default {
|
||||
width: 500,
|
||||
height: 500,
|
||||
color_bg: "#000000",
|
||||
color_text: "#ffffff"
|
||||
color_text: "#ffffff",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
@ -68,11 +96,9 @@ export default {
|
||||
const x = (this.toolData.width - textWidth) / 2;
|
||||
const y = this.toolData.height / 2 + 10;
|
||||
ctx.fillText(text, x, y);
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,24 +1,32 @@
|
||||
<template>
|
||||
<h2 class="tool-title">IBAN generator</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="style">Country</label>
|
||||
<div>
|
||||
<select id="style" v-model="toolData.country" v-on:change="result">
|
||||
<option value="">Select Country</option>
|
||||
<option v-for="(structure, code) in ibanStructure" :key="code" :value="code">
|
||||
<option
|
||||
v-for="(structure, code) in ibanStructure"
|
||||
:key="code"
|
||||
:value="code"
|
||||
>
|
||||
{{ structure.country }}
|
||||
</option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -34,8 +42,8 @@ function generateRandomNumber(length) {
|
||||
}
|
||||
|
||||
function generateRandomString(length) {
|
||||
const characters = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
|
||||
let result = '';
|
||||
const characters = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
|
||||
let result = "";
|
||||
for (let i = 0; i < length; i++) {
|
||||
result += characters.charAt(Math.floor(Math.random() * characters.length));
|
||||
}
|
||||
@ -45,26 +53,31 @@ function generateRandomString(length) {
|
||||
function mod97(string) {
|
||||
let checksum = string.slice(0, 2);
|
||||
for (let i = 2; i < string.length; i += 7) {
|
||||
checksum = (parseInt(checksum + string.slice(i, i + 7), 10) % 97).toString().padStart(2, '0');
|
||||
checksum = (parseInt(checksum + string.slice(i, i + 7), 10) % 97)
|
||||
.toString()
|
||||
.padStart(2, "0");
|
||||
}
|
||||
return parseInt(checksum);
|
||||
}
|
||||
|
||||
function calculateCheckDigits(iban) {
|
||||
const rearranged = iban.slice(4) + iban.slice(0, 4);
|
||||
const digits = rearranged.split('').map(char => {
|
||||
if (char >= 'A' && char <= 'Z') {
|
||||
return (char.charCodeAt(0) - 55).toString();
|
||||
}
|
||||
return char;
|
||||
}).join('');
|
||||
const digits = rearranged
|
||||
.split("")
|
||||
.map((char) => {
|
||||
if (char >= "A" && char <= "Z") {
|
||||
return (char.charCodeAt(0) - 55).toString();
|
||||
}
|
||||
return char;
|
||||
})
|
||||
.join("");
|
||||
const remainder = mod97(digits);
|
||||
return ('0' + (98 - remainder)).slice(-2);
|
||||
return ("0" + (98 - remainder)).slice(-2);
|
||||
}
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
@ -74,9 +87,17 @@ export default {
|
||||
toolResult: "",
|
||||
ibanStructure: {
|
||||
AD: { country: "Andorra", length: 24, structure: "F04F04F12" },
|
||||
AE: { country: "United Arab Emirates", length: 23, structure: "F03F16" },
|
||||
AE: {
|
||||
country: "United Arab Emirates",
|
||||
length: 23,
|
||||
structure: "F03F16",
|
||||
},
|
||||
AT: { country: "Austria", length: 20, structure: "F05F11" },
|
||||
BA: { country: "Bosnia and Herzegovina", length: 20, structure: "F03F03F08F02" },
|
||||
BA: {
|
||||
country: "Bosnia and Herzegovina",
|
||||
length: 20,
|
||||
structure: "F03F03F08F02",
|
||||
},
|
||||
BE: { country: "Belgium", length: 16, structure: "F03F07F02" },
|
||||
BG: { country: "Bulgaria", length: 22, structure: "U04F04F02F08" },
|
||||
BH: { country: "Bahrain", length: 22, structure: "U04F14" },
|
||||
@ -114,7 +135,11 @@ export default {
|
||||
MK: { country: "North Macedonia", length: 19, structure: "F03U10F02" },
|
||||
MR: { country: "Mauritania", length: 27, structure: "F05F05F11F02" },
|
||||
MT: { country: "Malta", length: 31, structure: "U04F05U18" },
|
||||
MU: { country: "Mauritius", length: 30, structure: "U04F02F02F12F03U03" },
|
||||
MU: {
|
||||
country: "Mauritius",
|
||||
length: 30,
|
||||
structure: "U04F02F02F12F03U03",
|
||||
},
|
||||
NL: { country: "Netherlands", length: 18, structure: "U04F10" },
|
||||
NO: { country: "Norway", length: 15, structure: "F04F06F01" },
|
||||
PK: { country: "Pakistan", length: 24, structure: "U04U16" },
|
||||
@ -137,17 +162,17 @@ export default {
|
||||
methods: {
|
||||
result() {
|
||||
if (!this.toolData.country.length) {
|
||||
this.toolResult = 'Select a country';
|
||||
this.toolResult = "Select a country";
|
||||
return;
|
||||
}
|
||||
|
||||
const structure = this.ibanStructure[this.toolData.country];
|
||||
if (!structure) {
|
||||
this.toolResult = 'Unsupported country';
|
||||
this.toolResult = "Unsupported country";
|
||||
return;
|
||||
}
|
||||
|
||||
let iban = this.toolData.country + '00'; // Add placeholder check digits
|
||||
let iban = this.toolData.country + "00"; // Add placeholder check digits
|
||||
const parts = structure.structure.match(/([FU])(\d+)/g);
|
||||
|
||||
for (const part of parts) {
|
||||
@ -155,10 +180,10 @@ export default {
|
||||
const length = parseInt(part.slice(1));
|
||||
|
||||
switch (type) {
|
||||
case 'F':
|
||||
case "F":
|
||||
iban += generateRandomNumber(length);
|
||||
break;
|
||||
case 'U':
|
||||
case "U":
|
||||
iban += generateRandomString(length);
|
||||
break;
|
||||
}
|
||||
@ -169,11 +194,9 @@ export default {
|
||||
iban = iban.slice(0, 2) + checkDigits + iban.slice(4);
|
||||
|
||||
this.toolResult = iban;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,34 +1,59 @@
|
||||
<template>
|
||||
<h2 class="tool-title">QR code</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<!-- Type options -->
|
||||
<div class="input-group">
|
||||
<label for="text">Data:</label>
|
||||
<textarea id="data" v-model="toolData.options.text" v-on:keyup="result"></textarea>
|
||||
<textarea
|
||||
id="data"
|
||||
v-model="toolData.options.text"
|
||||
v-on:keyup="result"
|
||||
></textarea>
|
||||
</div>
|
||||
|
||||
<!-- Basic options -->
|
||||
<div class="input-group">
|
||||
<label for="width">Width:</label>
|
||||
<input id="width" v-model.number="toolData.options.width" v-on:change="result" v-on:keyup="result" type="number">
|
||||
<input
|
||||
id="width"
|
||||
v-model.number="toolData.options.width"
|
||||
v-on:change="result"
|
||||
v-on:keyup="result"
|
||||
type="number"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="height">Height:</label>
|
||||
<input id="height" v-model.number="toolData.options.height" v-on:change="result" v-on:keyup="result" type="number">
|
||||
<input
|
||||
id="height"
|
||||
v-model.number="toolData.options.height"
|
||||
v-on:change="result"
|
||||
v-on:keyup="result"
|
||||
type="number"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="height">White border:</label>
|
||||
<input id="quietZone" v-model.number="toolData.options.quietZone" v-on:change="result" v-on:keyup="result"
|
||||
type="number">
|
||||
<input
|
||||
id="quietZone"
|
||||
v-model.number="toolData.options.quietZone"
|
||||
v-on:change="result"
|
||||
v-on:keyup="result"
|
||||
type="number"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="correctLevel">Correct Level:</label>
|
||||
<div>
|
||||
<select id="correctLevel" v-model="toolData.options.correctLevel" v-on:change="result">
|
||||
<select
|
||||
id="correctLevel"
|
||||
v-model="toolData.options.correctLevel"
|
||||
v-on:change="result"
|
||||
>
|
||||
<option value="L">L</option>
|
||||
<option value="M">M</option>
|
||||
<option value="Q">Q</option>
|
||||
@ -41,7 +66,13 @@
|
||||
<div class="input-group">
|
||||
<label for="logoUpload">Logo Upload:</label>
|
||||
<div>
|
||||
<input type="file" id="logoUpload" ref="logoUpload" @change="handleLogoUpload" accept="image/*">
|
||||
<input
|
||||
type="file"
|
||||
id="logoUpload"
|
||||
ref="logoUpload"
|
||||
@change="handleLogoUpload"
|
||||
accept="image/*"
|
||||
/>
|
||||
<button @click="removeLogo">🞩</button>
|
||||
</div>
|
||||
</div>
|
||||
@ -50,16 +81,29 @@
|
||||
<div class="input-group">
|
||||
<label for="backgroundImageUpload">Background Image Upload:</label>
|
||||
<div>
|
||||
<input type="file" id="backgroundImageUpload" ref="backgroundImageUpload" @change="handleBackgroundImageUpload"
|
||||
accept="image/*">
|
||||
<input
|
||||
type="file"
|
||||
id="backgroundImageUpload"
|
||||
ref="backgroundImageUpload"
|
||||
@change="handleBackgroundImageUpload"
|
||||
accept="image/*"
|
||||
/>
|
||||
<button @click="removeBackgroundImage">🞩</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="backgroundImageAlpha">Background Image Alpha:</label>
|
||||
<input id="backgroundImageAlpha" v-model.number="toolData.options.backgroundImageAlpha" step="0.1" min="0" max="1"
|
||||
v-on:change="result" v-on:keyup="result" type="number">
|
||||
<input
|
||||
id="backgroundImageAlpha"
|
||||
v-model.number="toolData.options.backgroundImageAlpha"
|
||||
step="0.1"
|
||||
min="0"
|
||||
max="1"
|
||||
v-on:change="result"
|
||||
v-on:keyup="result"
|
||||
type="number"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<!--
|
||||
@ -82,7 +126,7 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label>Result</label>
|
||||
@ -110,9 +154,9 @@ export default {
|
||||
backgroundImage: "",
|
||||
backgroundImageAlpha: 1,
|
||||
autoColor: true,
|
||||
drawer: "canvas"
|
||||
}
|
||||
}
|
||||
drawer: "canvas",
|
||||
},
|
||||
},
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
@ -129,10 +173,10 @@ export default {
|
||||
|
||||
// convert correction levels
|
||||
options.correctLevel = {
|
||||
"L": QRCode.CorrectLevel.L,
|
||||
"M": QRCode.CorrectLevel.M,
|
||||
"Q": QRCode.CorrectLevel.Q,
|
||||
"H": QRCode.CorrectLevel.H
|
||||
L: QRCode.CorrectLevel.L,
|
||||
M: QRCode.CorrectLevel.M,
|
||||
Q: QRCode.CorrectLevel.Q,
|
||||
H: QRCode.CorrectLevel.H,
|
||||
}[options.correctLevel];
|
||||
|
||||
console.log(options.correctLevel);
|
||||
@ -171,11 +215,9 @@ export default {
|
||||
removeBackgroundImage() {
|
||||
this.toolData.options.backgroundImage = "";
|
||||
this.result();
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Go JSON to struct</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -9,20 +9,36 @@
|
||||
|
||||
<div class="input-group">
|
||||
<div>
|
||||
<input id="flatten" name="flatten" v-model="toolData.flatten" v-on:change="result" type="checkbox"> <label
|
||||
for="flatten">inline type definitions</label>
|
||||
<input
|
||||
id="flatten"
|
||||
name="flatten"
|
||||
v-model="toolData.flatten"
|
||||
v-on:change="result"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="flatten">inline type definitions</label>
|
||||
</div>
|
||||
<div>
|
||||
<input id="omitempty" name="omitempty" v-model="toolData.omitempty" v-on:change="result" type="checkbox"> <label
|
||||
for="omitempty">omitempty</label>
|
||||
<input
|
||||
id="omitempty"
|
||||
name="omitempty"
|
||||
v-model="toolData.omitempty"
|
||||
v-on:change="result"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="omitempty">omitempty</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="json" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="json"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -30,7 +46,13 @@
|
||||
import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
// https://github.com/mholt/json-to-go
|
||||
function jsonToGo(json, typename, flatten = true, example = false, allOmitempty = false) {
|
||||
function jsonToGo(
|
||||
json,
|
||||
typename,
|
||||
flatten = true,
|
||||
example = false,
|
||||
allOmitempty = false
|
||||
) {
|
||||
let data;
|
||||
let scope;
|
||||
let go = "";
|
||||
@ -48,7 +70,7 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
} catch (e) {
|
||||
return {
|
||||
go: "",
|
||||
error: e.message
|
||||
error: e.message,
|
||||
};
|
||||
}
|
||||
|
||||
@ -58,12 +80,9 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
parseScope(scope);
|
||||
|
||||
return {
|
||||
go: flatten
|
||||
? go += accumulator
|
||||
: go
|
||||
go: flatten ? (go += accumulator) : go,
|
||||
};
|
||||
|
||||
|
||||
function parseScope(scope, depth = 0) {
|
||||
if (typeof scope === "object" && scope !== null) {
|
||||
if (Array.isArray(scope)) {
|
||||
@ -72,23 +91,20 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
for (let i = 0; i < scopeLength; i++) {
|
||||
const thisType = goType(scope[i]);
|
||||
if (!sliceType)
|
||||
sliceType = thisType;
|
||||
if (!sliceType) sliceType = thisType;
|
||||
else if (sliceType != thisType) {
|
||||
sliceType = mostSpecificPossibleGoType(thisType, sliceType);
|
||||
if (sliceType == "any")
|
||||
break;
|
||||
if (sliceType == "any") break;
|
||||
}
|
||||
}
|
||||
|
||||
const slice = flatten && ["struct", "slice"].includes(sliceType)
|
||||
? `[]${parent}`
|
||||
: `[]`;
|
||||
const slice =
|
||||
flatten && ["struct", "slice"].includes(sliceType)
|
||||
? `[]${parent}`
|
||||
: `[]`;
|
||||
|
||||
if (flatten && depth >= 2)
|
||||
appender(slice);
|
||||
else
|
||||
append(slice);
|
||||
if (flatten && depth >= 2) appender(slice);
|
||||
else append(slice);
|
||||
if (sliceType == "struct") {
|
||||
const allFields = {};
|
||||
|
||||
@ -100,7 +116,7 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
if (!(keyname in allFields)) {
|
||||
allFields[keyname] = {
|
||||
value: scope[i][keyname],
|
||||
count: 0
|
||||
count: 0,
|
||||
};
|
||||
} else {
|
||||
const existingValue = allFields[keyname].value;
|
||||
@ -115,7 +131,7 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
keyname = `${keyname}_${uuidv4()}`;
|
||||
allFields[keyname] = {
|
||||
value: currentValue,
|
||||
count: 0
|
||||
count: 0,
|
||||
};
|
||||
}
|
||||
}
|
||||
@ -126,9 +142,12 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
// create a common struct with all fields found in the current array
|
||||
// omitempty dict indicates if a field is optional
|
||||
const keys = Object.keys(allFields), struct = {}, omitempty = {};
|
||||
const keys = Object.keys(allFields),
|
||||
struct = {},
|
||||
omitempty = {};
|
||||
for (let k in keys) {
|
||||
const keyname = keys[k], elem = allFields[keyname];
|
||||
const keyname = keys[k],
|
||||
elem = allFields[keyname];
|
||||
|
||||
struct[keyname] = elem.value;
|
||||
omitempty[keyname] = elem.count != scopeLength;
|
||||
@ -164,11 +183,7 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
function parseStruct(depth, innerTabs, scope, omitempty) {
|
||||
if (flatten) {
|
||||
stack.push(
|
||||
depth >= 2
|
||||
? "\n"
|
||||
: ""
|
||||
);
|
||||
stack.push(depth >= 2 ? "\n" : "");
|
||||
}
|
||||
|
||||
const seenTypeNames = [];
|
||||
@ -198,11 +213,11 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
appender(typename + " ");
|
||||
parent = typename;
|
||||
parseScope(scope[keys[i]], depth);
|
||||
appender(" `json:\"" + keyname);
|
||||
appender(' `json:"' + keyname);
|
||||
if (allOmitempty || (omitempty && omitempty[keys[i]] === true)) {
|
||||
appender(",omitempty");
|
||||
}
|
||||
appender("\"`\n");
|
||||
appender('"`\n');
|
||||
}
|
||||
indenter(--innerTabs);
|
||||
appender("}");
|
||||
@ -218,25 +233,27 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
append(typename + " ");
|
||||
parent = typename;
|
||||
parseScope(scope[keys[i]], depth);
|
||||
append(" `json:\"" + keyname);
|
||||
append(' `json:"' + keyname);
|
||||
if (allOmitempty || (omitempty && omitempty[keys[i]] === true)) {
|
||||
append(",omitempty");
|
||||
}
|
||||
if (example && scope[keys[i]] !== "" && typeof scope[keys[i]] !== "object") {
|
||||
append("\" example:\"" + scope[keys[i]]);
|
||||
if (
|
||||
example &&
|
||||
scope[keys[i]] !== "" &&
|
||||
typeof scope[keys[i]] !== "object"
|
||||
) {
|
||||
append('" example:"' + scope[keys[i]]);
|
||||
}
|
||||
append("\"`\n");
|
||||
append('"`\n');
|
||||
}
|
||||
indent(--tabs);
|
||||
append("}");
|
||||
}
|
||||
if (flatten)
|
||||
accumulator += stack.pop();
|
||||
if (flatten) accumulator += stack.pop();
|
||||
}
|
||||
|
||||
function indent(tabs) {
|
||||
for (let i = 0; i < tabs; i++)
|
||||
go += "\t";
|
||||
for (let i = 0; i < tabs; i++) go += "\t";
|
||||
}
|
||||
|
||||
function append(str) {
|
||||
@ -244,8 +261,7 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
}
|
||||
|
||||
function indenter(tabs) {
|
||||
for (let i = 0; i < tabs; i++)
|
||||
stack[stack.length - 1] += "\t";
|
||||
for (let i = 0; i < tabs; i++) stack[stack.length - 1] += "\t";
|
||||
}
|
||||
|
||||
function appender(str) {
|
||||
@ -274,7 +290,7 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
function format(str) {
|
||||
str = formatNumber(str);
|
||||
|
||||
let sanitized = toProperCase(str).replace(/[^a-z0-9]/ig, "");
|
||||
let sanitized = toProperCase(str).replace(/[^a-z0-9]/gi, "");
|
||||
if (!sanitized) {
|
||||
return "NAMING_FAILED";
|
||||
}
|
||||
@ -286,15 +302,20 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
// Adds a prefix to a number to make an appropriate identifier in Go
|
||||
function formatNumber(str) {
|
||||
if (!str)
|
||||
return "";
|
||||
else if (str.match(/^\d+$/))
|
||||
str = "Num" + str;
|
||||
if (!str) return "";
|
||||
else if (str.match(/^\d+$/)) str = "Num" + str;
|
||||
else if (str.charAt(0).match(/\d/)) {
|
||||
const numbers = {
|
||||
"0": "Zero_", "1": "One_", "2": "Two_", "3": "Three_",
|
||||
"4": "Four_", "5": "Five_", "6": "Six_", "7": "Seven_",
|
||||
"8": "Eight_", "9": "Nine_"
|
||||
0: "Zero_",
|
||||
1: "One_",
|
||||
2: "Two_",
|
||||
3: "Three_",
|
||||
4: "Four_",
|
||||
5: "Five_",
|
||||
6: "Six_",
|
||||
7: "Seven_",
|
||||
8: "Eight_",
|
||||
9: "Nine_",
|
||||
};
|
||||
str = numbers[str.charAt(0)] + str.substr(1);
|
||||
}
|
||||
@ -304,28 +325,22 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
// Determines the most appropriate Go type
|
||||
function goType(val) {
|
||||
if (val === null)
|
||||
return "any";
|
||||
if (val === null) return "any";
|
||||
|
||||
switch (typeof val) {
|
||||
case "string":
|
||||
if (/^\d{4}-\d\d-\d\dT\d\d:\d\d:\d\d(\.\d+)?(\+\d\d:\d\d|Z)$/.test(val))
|
||||
return "time.Time";
|
||||
else
|
||||
return "string";
|
||||
else return "string";
|
||||
case "number":
|
||||
if (val % 1 === 0) {
|
||||
if (val > -2147483648 && val < 2147483647)
|
||||
return "int";
|
||||
else
|
||||
return "int64";
|
||||
} else
|
||||
return "float64";
|
||||
if (val > -2147483648 && val < 2147483647) return "int";
|
||||
else return "int64";
|
||||
} else return "float64";
|
||||
case "boolean":
|
||||
return "bool";
|
||||
case "object":
|
||||
if (Array.isArray(val))
|
||||
return "slice";
|
||||
if (Array.isArray(val)) return "slice";
|
||||
return "struct";
|
||||
default:
|
||||
return "any";
|
||||
@ -334,14 +349,10 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
// Given two types, returns the more specific of the two
|
||||
function mostSpecificPossibleGoType(typ1, typ2) {
|
||||
if (typ1.substr(0, 5) == "float"
|
||||
&& typ2.substr(0, 3) == "int")
|
||||
return typ1;
|
||||
else if (typ1.substr(0, 3) == "int"
|
||||
&& typ2.substr(0, 5) == "float")
|
||||
if (typ1.substr(0, 5) == "float" && typ2.substr(0, 3) == "int") return typ1;
|
||||
else if (typ1.substr(0, 3) == "int" && typ2.substr(0, 5) == "float")
|
||||
return typ2;
|
||||
else
|
||||
return "any";
|
||||
else return "any";
|
||||
}
|
||||
|
||||
// Proper cases a string according to Go conventions
|
||||
@ -353,34 +364,73 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
// https://github.com/golang/lint/blob/5614ed5bae6fb75893070bdc0996a68765fdd275/lint.go#L771-L810
|
||||
const commonInitialisms = [
|
||||
"ACL", "API", "ASCII", "CPU", "CSS", "DNS", "EOF", "GUID", "HTML", "HTTP",
|
||||
"HTTPS", "ID", "IP", "JSON", "LHS", "QPS", "RAM", "RHS", "RPC", "SLA",
|
||||
"SMTP", "SQL", "SSH", "TCP", "TLS", "TTL", "UDP", "UI", "UID", "UUID",
|
||||
"URI", "URL", "UTF8", "VM", "XML", "XMPP", "XSRF", "XSS"
|
||||
"ACL",
|
||||
"API",
|
||||
"ASCII",
|
||||
"CPU",
|
||||
"CSS",
|
||||
"DNS",
|
||||
"EOF",
|
||||
"GUID",
|
||||
"HTML",
|
||||
"HTTP",
|
||||
"HTTPS",
|
||||
"ID",
|
||||
"IP",
|
||||
"JSON",
|
||||
"LHS",
|
||||
"QPS",
|
||||
"RAM",
|
||||
"RHS",
|
||||
"RPC",
|
||||
"SLA",
|
||||
"SMTP",
|
||||
"SQL",
|
||||
"SSH",
|
||||
"TCP",
|
||||
"TLS",
|
||||
"TTL",
|
||||
"UDP",
|
||||
"UI",
|
||||
"UID",
|
||||
"UUID",
|
||||
"URI",
|
||||
"URL",
|
||||
"UTF8",
|
||||
"VM",
|
||||
"XML",
|
||||
"XMPP",
|
||||
"XSRF",
|
||||
"XSS",
|
||||
];
|
||||
|
||||
return str.replace(/(^|[^a-zA-Z])([a-z]+)/g, function(unused, sep, frag) {
|
||||
if (commonInitialisms.indexOf(frag.toUpperCase()) >= 0)
|
||||
return sep + frag.toUpperCase();
|
||||
else
|
||||
return sep + frag[0].toUpperCase() + frag.substr(1).toLowerCase();
|
||||
}).replace(/([A-Z])([a-z]+)/g, function(unused, sep, frag) {
|
||||
if (commonInitialisms.indexOf(sep + frag.toUpperCase()) >= 0)
|
||||
return (sep + frag).toUpperCase();
|
||||
else
|
||||
return sep + frag;
|
||||
});
|
||||
return str
|
||||
.replace(/(^|[^a-zA-Z])([a-z]+)/g, function (unused, sep, frag) {
|
||||
if (commonInitialisms.indexOf(frag.toUpperCase()) >= 0)
|
||||
return sep + frag.toUpperCase();
|
||||
else return sep + frag[0].toUpperCase() + frag.substr(1).toLowerCase();
|
||||
})
|
||||
.replace(/([A-Z])([a-z]+)/g, function (unused, sep, frag) {
|
||||
if (commonInitialisms.indexOf(sep + frag.toUpperCase()) >= 0)
|
||||
return (sep + frag).toUpperCase();
|
||||
else return sep + frag;
|
||||
});
|
||||
}
|
||||
|
||||
function uuidv4() {
|
||||
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function(c) {
|
||||
var r = Math.random() * 16 | 0, v = c == "x" ? r : (r & 0x3 | 0x8);
|
||||
return v.toString(16);
|
||||
});
|
||||
return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(
|
||||
/[xy]/g,
|
||||
function (c) {
|
||||
var r = (Math.random() * 16) | 0,
|
||||
v = c == "x" ? r : (r & 0x3) | 0x8;
|
||||
return v.toString(16);
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
function getOriginalName(unique) {
|
||||
const reLiteralUUID = /^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;
|
||||
const reLiteralUUID =
|
||||
/^[0-9a-f]{8}-[0-9a-f]{4}-[1-5][0-9a-f]{3}-[89ab][0-9a-f]{3}-[0-9a-f]{12}$/i;
|
||||
const uuidLength = 36;
|
||||
|
||||
if (unique.length >= uuidLength) {
|
||||
@ -394,8 +444,10 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
function compareObjects(objectA, objectB) {
|
||||
const object = "[object Object]";
|
||||
return Object.prototype.toString.call(objectA) === object
|
||||
&& Object.prototype.toString.call(objectB) === object;
|
||||
return (
|
||||
Object.prototype.toString.call(objectA) === object &&
|
||||
Object.prototype.toString.call(objectB) === object
|
||||
);
|
||||
}
|
||||
|
||||
function compareObjectKeys(itemAKeys, itemBKeys) {
|
||||
@ -403,16 +455,13 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
const lengthB = itemBKeys.length;
|
||||
|
||||
// nothing to compare, probably identical
|
||||
if (lengthA == 0 && lengthB == 0)
|
||||
return true;
|
||||
if (lengthA == 0 && lengthB == 0) return true;
|
||||
|
||||
// duh
|
||||
if (lengthA != lengthB)
|
||||
return false;
|
||||
if (lengthA != lengthB) return false;
|
||||
|
||||
for (let item of itemAKeys) {
|
||||
if (!itemBKeys.includes(item))
|
||||
return false;
|
||||
if (!itemBKeys.includes(item)) return false;
|
||||
}
|
||||
return true;
|
||||
}
|
||||
@ -427,16 +476,16 @@ function jsonToGo(json, typename, flatten = true, example = false, allOmitempty
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
flatten: false,
|
||||
omitempty: false
|
||||
omitempty: false,
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -446,18 +495,22 @@ export default {
|
||||
return;
|
||||
}
|
||||
|
||||
const json = jsonToGo(this.toolData.data, "", this.toolData.flatten, false, this.toolData.omitempty);
|
||||
const json = jsonToGo(
|
||||
this.toolData.data,
|
||||
"",
|
||||
this.toolData.flatten,
|
||||
false,
|
||||
this.toolData.omitempty
|
||||
);
|
||||
if (json.error?.length) {
|
||||
this.toolResult = "invalid json";
|
||||
return;
|
||||
}
|
||||
|
||||
this.toolResult = json.go;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,27 +1,42 @@
|
||||
<template>
|
||||
<h2 class="tool-title">SQL tables Go struct</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">MariaDB</label>
|
||||
<MonacoEditor name="dialects_mariadb" language="sql" :value="toolData.dialects.mariadb" readonly="true"
|
||||
style="height: 500px"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="dialects_mariadb"
|
||||
language="sql"
|
||||
:value="toolData.dialects.mariadb"
|
||||
readonly="true"
|
||||
style="height: 500px"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">PostgreSQL</label>
|
||||
<MonacoEditor name="dialects_postgresql" language="sql" :value="toolData.dialects.postgresql" readonly="true"
|
||||
style="height: 500px"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="dialects_postgresql"
|
||||
language="sql"
|
||||
:value="toolData.dialects.postgresql"
|
||||
readonly="true"
|
||||
style="height: 500px"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">OracleSQL</label>
|
||||
<MonacoEditor name="dialects_oraclesql" language="sql" :value="toolData.dialects.oraclesql" readonly="true"
|
||||
style="height: 500px"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="dialects_oraclesql"
|
||||
language="sql"
|
||||
:value="toolData.dialects.oraclesql"
|
||||
readonly="true"
|
||||
style="height: 500px"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -30,7 +45,7 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
@ -137,14 +152,12 @@ export default {
|
||||
--WHERE type_info = 'UNKNOWN'
|
||||
GROUP BY table_name)
|
||||
--WHERE table_name = 'V8EventVw'
|
||||
ORDER BY table_name`
|
||||
}
|
||||
}
|
||||
ORDER BY table_name`,
|
||||
},
|
||||
},
|
||||
};
|
||||
}
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">JSON formatter</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<textarea id="data" v-model="toolData.data" v-on:keyup="result"></textarea>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="json" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="json"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -20,14 +24,14 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: ""
|
||||
data: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -36,16 +40,18 @@ export default {
|
||||
this.toolResult = "";
|
||||
} else {
|
||||
try {
|
||||
this.toolResult = JSON.stringify(JSON.parse(this.toolData.data), null, 4);
|
||||
this.toolResult = JSON.stringify(
|
||||
JSON.parse(this.toolData.data),
|
||||
null,
|
||||
4
|
||||
);
|
||||
} catch (e) {
|
||||
this.toolResult = "invalid syntax";
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">JSON Minifier</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<textarea id="data" v-model="toolData.data" v-on:keyup="minify"></textarea>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="json" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="json"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -20,14 +24,14 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: ""
|
||||
data: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -41,11 +45,9 @@ export default {
|
||||
this.toolResult = "invalid syntax";
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">PHP array to Json</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -9,16 +9,26 @@
|
||||
|
||||
<div class="input-group">
|
||||
<div>
|
||||
<input id="minify_json" name="minify_json" v-model="toolData.minifyJson" v-on:change="result" type="checkbox">
|
||||
<input
|
||||
id="minify_json"
|
||||
name="minify_json"
|
||||
v-model="toolData.minifyJson"
|
||||
v-on:change="result"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="minify_json">minify json</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="json" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="json"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -29,33 +39,32 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
minifyJson: ""
|
||||
minifyJson: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
axios.post(`${config.APP_URL}/api/php/php_array_to_json.php`, {
|
||||
data: this.toolData.data,
|
||||
minify_json: this.toolData.minifyJson ? true : false
|
||||
})
|
||||
axios
|
||||
.post(`${config.APP_URL}/api/php/php_array_to_json.php`, {
|
||||
data: this.toolData.data,
|
||||
minify_json: !!this.toolData.minifyJson,
|
||||
})
|
||||
.then((response) => {
|
||||
this.toolResult = response.data.data;
|
||||
|
||||
Promise.resolve(response);
|
||||
});
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">PHP serialize</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -11,18 +11,36 @@
|
||||
<label for="data_delimiter">Mode</label>
|
||||
|
||||
<div>
|
||||
<input id="mode_serialize" value="serialize" name="mode" v-model="toolData.mode" v-on:change="result"
|
||||
type="radio"> <label for="mode_serialize">serialize</label><br>
|
||||
<input id="mode_unserialize" value="unserialize" name="mode" v-model="toolData.mode" v-on:change="result"
|
||||
type="radio"> <label for="mode_unserialize">unserialize</label><br>
|
||||
<input
|
||||
id="mode_serialize"
|
||||
value="serialize"
|
||||
name="mode"
|
||||
v-model="toolData.mode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="mode_serialize">serialize</label><br />
|
||||
<input
|
||||
id="mode_unserialize"
|
||||
value="unserialize"
|
||||
name="mode"
|
||||
v-model="toolData.mode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="mode_unserialize">unserialize</label><br />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="json" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="json"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -33,33 +51,32 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
mode: "serialize"
|
||||
mode: "serialize",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
axios.post(`${config.APP_URL}/api/php/php_serialize.php`, {
|
||||
data: this.toolData.data,
|
||||
mode: this.toolData.mode
|
||||
})
|
||||
axios
|
||||
.post(`${config.APP_URL}/api/php/php_serialize.php`, {
|
||||
data: this.toolData.data,
|
||||
mode: this.toolData.mode,
|
||||
})
|
||||
.then((response) => {
|
||||
this.toolResult = response.data;
|
||||
|
||||
Promise.resolve(response);
|
||||
});
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str to PHP array</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -9,20 +9,36 @@
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data_delimiter">Delimiter</label>
|
||||
<input id="data_delimiter" v-model="toolData.dataDelimiter" placeholder="," type="text" v-on:keyup="result"
|
||||
:disabled="toolData.dataIsJson">
|
||||
<input
|
||||
id="data_delimiter"
|
||||
v-model="toolData.dataDelimiter"
|
||||
placeholder=","
|
||||
type="text"
|
||||
v-on:keyup="result"
|
||||
:disabled="toolData.dataIsJson"
|
||||
/>
|
||||
|
||||
<div>
|
||||
<input id="data_is_json" name="data_is_json" v-model="toolData.dataIsJson" v-on:change="result" type="checkbox">
|
||||
<input
|
||||
id="data_is_json"
|
||||
name="data_is_json"
|
||||
v-model="toolData.dataIsJson"
|
||||
v-on:change="result"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="data_is_json">is json</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="php" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="php"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -33,35 +49,34 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
dataDelimiter: "",
|
||||
dataIsJson: false
|
||||
dataIsJson: false,
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
axios.post(`${config.APP_URL}/api/php/str_to_php_array.php`, {
|
||||
data: this.toolData.data,
|
||||
delimiter: this.toolData.dataDelimiter,
|
||||
mode: this.toolData.dataIsJson ? "json" : null
|
||||
})
|
||||
axios
|
||||
.post(`${config.APP_URL}/api/php/str_to_php_array.php`, {
|
||||
data: this.toolData.data,
|
||||
delimiter: this.toolData.dataDelimiter,
|
||||
mode: this.toolData.dataIsJson ? "json" : null,
|
||||
})
|
||||
.then((response) => {
|
||||
this.toolResult = response.data;
|
||||
|
||||
Promise.resolve(response);
|
||||
});
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">SQL formatter</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<textarea id="data" v-model="toolData.data" v-on:keyup="result"></textarea>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="sql" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="sql"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -21,15 +25,15 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
minifyJson: ""
|
||||
minifyJson: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -46,20 +50,18 @@ export default {
|
||||
custom: [
|
||||
{ regex: String.raw`\{\$?[\w\d\->\[\]]+\}` },
|
||||
{ regex: String.raw`\$[\w\d\->\[\]]+` },
|
||||
{ regex: String.raw`['"].*['"]` }
|
||||
{ regex: String.raw`['"].*['"]` },
|
||||
],
|
||||
positional: true,
|
||||
named: [":", "@"]
|
||||
}
|
||||
named: [":", "@"],
|
||||
},
|
||||
});
|
||||
} catch (e) {
|
||||
this.toolResult = "invalid syntax";
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,10 +1,16 @@
|
||||
<template>
|
||||
<h2 class="tool-title">SQL split IN</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="field_name">Field name</label>
|
||||
<input id="field_name" v-model="toolData.fieldName" v-on:keyup="result" placeholder="D.ID" type="text">
|
||||
<input
|
||||
id="field_name"
|
||||
v-model="toolData.fieldName"
|
||||
v-on:keyup="result"
|
||||
placeholder="D.ID"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
@ -14,32 +20,70 @@
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data_delimiter">Delimiter</label>
|
||||
<input id="data_delimiter" v-model="toolData.dataDelimiter" v-on:keyup="result" placeholder="," type="text">
|
||||
<input
|
||||
id="data_delimiter"
|
||||
v-model="toolData.dataDelimiter"
|
||||
v-on:keyup="result"
|
||||
placeholder=","
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label>Wrap in quotes</label>
|
||||
|
||||
<div>
|
||||
<input id="wrap_in_quotes_no" value="" name="wrap_in_quotes" v-model="toolData.wrapInQuotes" v-on:change="result"
|
||||
type="radio"> <label for="wrap_in_quotes_no">No</label><br>
|
||||
<input id="wrap_in_quotes_single" value="single" name="wrap_in_quotes" v-model="toolData.wrapInQuotes"
|
||||
v-on:change="result" type="radio"> <label for="wrap_in_quotes_single">Single</label><br>
|
||||
<input id="wrap_in_quotes_double" value="double" name="wrap_in_quotes" v-model="toolData.wrapInQuotes"
|
||||
v-on:change="result" type="radio"> <label for="wrap_in_quotes_double">Double</label>
|
||||
<input
|
||||
id="wrap_in_quotes_no"
|
||||
value=""
|
||||
name="wrap_in_quotes"
|
||||
v-model="toolData.wrapInQuotes"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="wrap_in_quotes_no">No</label><br />
|
||||
<input
|
||||
id="wrap_in_quotes_single"
|
||||
value="single"
|
||||
name="wrap_in_quotes"
|
||||
v-model="toolData.wrapInQuotes"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="wrap_in_quotes_single">Single</label><br />
|
||||
<input
|
||||
id="wrap_in_quotes_double"
|
||||
value="double"
|
||||
name="wrap_in_quotes"
|
||||
v-model="toolData.wrapInQuotes"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="wrap_in_quotes_double">Double</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="chunk_by">Chunk by</label>
|
||||
<input id="chunk_by" v-model="toolData.chunkBy" v-on:keyup="result" placeholder="900" min="1" type="number">
|
||||
<input
|
||||
id="chunk_by"
|
||||
v-model="toolData.chunkBy"
|
||||
v-on:keyup="result"
|
||||
placeholder="900"
|
||||
min="1"
|
||||
type="number"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="sql" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="sql"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -57,7 +101,7 @@ function arrayChunk(arr, chunkSize) {
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
@ -66,15 +110,16 @@ export default {
|
||||
data: "",
|
||||
dataDelimiter: "",
|
||||
wrapInQuotes: "",
|
||||
chunkBy: 900
|
||||
chunkBy: 900,
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
let data = this.toolData.data
|
||||
.split(this.toolData.dataDelimiter ? this.toolData.dataDelimiter : "\n");
|
||||
let data = this.toolData.data.split(
|
||||
this.toolData.dataDelimiter ? this.toolData.dataDelimiter : "\n"
|
||||
);
|
||||
|
||||
for (let i = 0; i < data.length; i++) {
|
||||
let value = data[i];
|
||||
@ -84,7 +129,7 @@ export default {
|
||||
value = "'" + value.replace("'", "\\'") + "'";
|
||||
break;
|
||||
case "double":
|
||||
value = "\"" + value.replace("\"", "\\\"") + "\"";
|
||||
value = '"' + value.replace('"', '\\"') + '"';
|
||||
break;
|
||||
}
|
||||
|
||||
@ -100,12 +145,15 @@ export default {
|
||||
data[i] = "(" + chunk.join(",") + ")";
|
||||
}
|
||||
|
||||
this.toolResult = "(" + this.toolData.fieldName + " IN " + data.join(" OR " + this.toolData.fieldName + " IN ") + ")";
|
||||
}
|
||||
}
|
||||
this.toolResult =
|
||||
"(" +
|
||||
this.toolData.fieldName +
|
||||
" IN " +
|
||||
data.join(" OR " + this.toolData.fieldName + " IN ") +
|
||||
")";
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Fix ru-en keyboard</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -11,18 +11,36 @@
|
||||
<label>Mode</label>
|
||||
|
||||
<div>
|
||||
<input id="keyboard_mode_en_ru" value="en-ru" name="keyboard_mode" v-model="toolData.keyboardMode"
|
||||
v-on:change="result" type="radio"> <label for="keyboard_mode_en_ru">en -> ru</label><br>
|
||||
<input id="keyboard_mode_ru_en" value="ru-en" name="keyboard_mode" v-model="toolData.keyboardMode"
|
||||
v-on:change="result" type="radio"> <label for="keyboard_mode_ru_en">ru -> en</label>
|
||||
<input
|
||||
id="keyboard_mode_en_ru"
|
||||
value="en-ru"
|
||||
name="keyboard_mode"
|
||||
v-model="toolData.keyboardMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="keyboard_mode_en_ru">en -> ru</label><br />
|
||||
<input
|
||||
id="keyboard_mode_ru_en"
|
||||
value="ru-en"
|
||||
name="keyboard_mode"
|
||||
v-model="toolData.keyboardMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="keyboard_mode_ru_en">ru -> en</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -32,19 +50,21 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
keyboardMode: "en-ru"
|
||||
keyboardMode: "en-ru",
|
||||
},
|
||||
dictionary: {
|
||||
"en-ru": "`~!@#$%^&qwertyuiop[]asdfghjkl;'zxcvbnm,./QWERTYUIOP{}ASDFGHJKL:\"ZXCVBNM<>?",
|
||||
"ru-en": "ёЁ!\"№;%:?йцукенгшщзхъфывапролджэячсмитьбю.ЙЦУКЕНГШЩЗХЪФЫВАПРОЛДЖЭЯЧСМИТЬБЮ,"
|
||||
"en-ru":
|
||||
"`~!@#$%^&qwertyuiop[]asdfghjkl;'zxcvbnm,./QWERTYUIOP{}ASDFGHJKL:\"ZXCVBNM<>?",
|
||||
"ru-en":
|
||||
'ёЁ!"№;%:?йцукенгшщзхъфывапролджэячсмитьбю.ЙЦУКЕНГШЩЗХЪФЫВАПРОЛДЖЭЯЧСМИТЬБЮ,',
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -52,7 +72,10 @@ export default {
|
||||
let data = unproxy(this.toolData.data).split("");
|
||||
|
||||
const dict = this.dictionary[this.toolData.keyboardMode];
|
||||
const dictOpposite = this.dictionary[this.toolData.keyboardMode === "en-ru" ? "ru-en" : "en-ru"];
|
||||
const dictOpposite =
|
||||
this.dictionary[
|
||||
this.toolData.keyboardMode === "en-ru" ? "ru-en" : "en-ru"
|
||||
];
|
||||
|
||||
for (let i = 0; i < data.length; i++) {
|
||||
const dictPos = dict.indexOf(data[i]);
|
||||
@ -62,11 +85,9 @@ export default {
|
||||
}
|
||||
|
||||
this.toolResult = data.join("");
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str length</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -9,24 +9,57 @@
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data_delimiter">Delimiter</label>
|
||||
<input id="data_delimiter" v-model="toolData.dataDelimiter" v-on:keyup="result" placeholder="," type="text">
|
||||
<input
|
||||
id="data_delimiter"
|
||||
v-model="toolData.dataDelimiter"
|
||||
v-on:keyup="result"
|
||||
placeholder=","
|
||||
type="text"
|
||||
/>
|
||||
|
||||
<div>
|
||||
<input id="count_mode_characters" value="characters" name="count_mode" v-model="toolData.countMode"
|
||||
v-on:change="result" :disabled="toolData.dataDelimiter.length" type="radio"> <label
|
||||
for="count_mode_characters">characters</label><br>
|
||||
<input id="count_mode_words" value="words" name="count_mode" v-model="toolData.countMode" v-on:change="result"
|
||||
:disabled="toolData.dataDelimiter.length" type="radio"> <label for="count_mode_words">words</label><br>
|
||||
<input id="count_mode_lines" value="lines" name="count_mode" v-model="toolData.countMode" v-on:change="result"
|
||||
:disabled="toolData.dataDelimiter.length" type="radio"> <label for="count_mode_lines">lines</label>
|
||||
<input
|
||||
id="count_mode_characters"
|
||||
value="characters"
|
||||
name="count_mode"
|
||||
v-model="toolData.countMode"
|
||||
v-on:change="result"
|
||||
:disabled="toolData.dataDelimiter.length"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="count_mode_characters">characters</label><br />
|
||||
<input
|
||||
id="count_mode_words"
|
||||
value="words"
|
||||
name="count_mode"
|
||||
v-model="toolData.countMode"
|
||||
v-on:change="result"
|
||||
:disabled="toolData.dataDelimiter.length"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="count_mode_words">words</label><br />
|
||||
<input
|
||||
id="count_mode_lines"
|
||||
value="lines"
|
||||
name="count_mode"
|
||||
v-model="toolData.countMode"
|
||||
v-on:change="result"
|
||||
:disabled="toolData.dataDelimiter.length"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="count_mode_lines">lines</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -35,16 +68,16 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
dataDelimiter: "",
|
||||
countMode: "characters"
|
||||
countMode: "characters",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -52,7 +85,9 @@ export default {
|
||||
if (!this.toolData.data.length) {
|
||||
this.toolResult = 0;
|
||||
} else if (this.toolData.dataDelimiter.length) {
|
||||
this.toolResult = this.toolData.data.split(this.toolData.dataDelimiter).length;
|
||||
this.toolResult = this.toolData.data.split(
|
||||
this.toolData.dataDelimiter
|
||||
).length;
|
||||
} else {
|
||||
switch (this.toolData.countMode) {
|
||||
case "characters":
|
||||
@ -66,11 +101,9 @@ export default {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str to NATO alphabet</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<input id="data" v-model="toolData.data" v-on:keyup="result" type="text">
|
||||
<input id="data" v-model="toolData.data" v-on:keyup="result" type="text" />
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="php" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="php"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -22,34 +26,55 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: ""
|
||||
data: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
const natoAlphabet = {
|
||||
"A": "Alpha", "B": "Bravo", "C": "Charlie", "D": "Delta",
|
||||
"E": "Echo", "F": "Foxtrot", "G": "Golf", "H": "Hotel",
|
||||
"I": "India", "J": "Juliett", "K": "Kilo", "L": "Lima",
|
||||
"M": "Mike", "N": "November", "O": "Oscar", "P": "Papa",
|
||||
"Q": "Quebec", "R": "Romeo", "S": "Sierra", "T": "Tango",
|
||||
"U": "Uniform", "V": "Victor", "W": "Whiskey", "X": "X-ray",
|
||||
"Y": "Yankee", "Z": "Zulu"
|
||||
A: "Alpha",
|
||||
B: "Bravo",
|
||||
C: "Charlie",
|
||||
D: "Delta",
|
||||
E: "Echo",
|
||||
F: "Foxtrot",
|
||||
G: "Golf",
|
||||
H: "Hotel",
|
||||
I: "India",
|
||||
J: "Juliett",
|
||||
K: "Kilo",
|
||||
L: "Lima",
|
||||
M: "Mike",
|
||||
N: "November",
|
||||
O: "Oscar",
|
||||
P: "Papa",
|
||||
Q: "Quebec",
|
||||
R: "Romeo",
|
||||
S: "Sierra",
|
||||
T: "Tango",
|
||||
U: "Uniform",
|
||||
V: "Victor",
|
||||
W: "Whiskey",
|
||||
X: "X-ray",
|
||||
Y: "Yankee",
|
||||
Z: "Zulu",
|
||||
};
|
||||
|
||||
this.toolResult = this.toolData.data.toUpperCase().split("").map(char => natoAlphabet[char] || char).join(" ");
|
||||
}
|
||||
}
|
||||
this.toolResult = this.toolData.data
|
||||
.toUpperCase()
|
||||
.split("")
|
||||
.map((char) => natoAlphabet[char] || char)
|
||||
.join(" ");
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str numeronym</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<input id="data" v-model="toolData.data" v-on:keyup="result" type="text">
|
||||
<input id="data" v-model="toolData.data" v-on:keyup="result" type="text" />
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -21,14 +25,14 @@ import { unproxy } from "../../utils/unproxy";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: ""
|
||||
data: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -45,11 +49,9 @@ export default {
|
||||
const numberOfLetters = word.length - 2;
|
||||
|
||||
this.toolResult = `${firstLetter}${numberOfLetters}${lastLetter}`;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str pad</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -9,21 +9,49 @@
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Symbol</label>
|
||||
<input id="data" class="input" v-model="toolData.padSymbol" v-on:keyup="result" placeholder="0" type="text">
|
||||
<input
|
||||
id="data"
|
||||
class="input"
|
||||
v-model="toolData.padSymbol"
|
||||
v-on:keyup="result"
|
||||
placeholder="0"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Pad count / Fixed length</label>
|
||||
<input id="data" class="input" v-model="toolData.padCount" v-on:keyup="result" placeholder="0" type="text">
|
||||
<input
|
||||
id="data"
|
||||
class="input"
|
||||
v-model="toolData.padCount"
|
||||
v-on:keyup="result"
|
||||
placeholder="0"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label>Pad position</label>
|
||||
|
||||
<div>
|
||||
<input id="pad_pos_begin" value="begin" name="pad_pos" v-model="toolData.padPos" v-on:change="result"
|
||||
type="radio"> <label for="pad_pos_begin">begin</label><br>
|
||||
<input id="pad_pos_end" value="end" name="pad_pos" v-model="toolData.padPos" v-on:change="result" type="radio">
|
||||
<input
|
||||
id="pad_pos_begin"
|
||||
value="begin"
|
||||
name="pad_pos"
|
||||
v-model="toolData.padPos"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="pad_pos_begin">begin</label><br />
|
||||
<input
|
||||
id="pad_pos_end"
|
||||
value="end"
|
||||
name="pad_pos"
|
||||
v-model="toolData.padPos"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="pad_pos_end">end</label>
|
||||
</div>
|
||||
</div>
|
||||
@ -32,18 +60,36 @@
|
||||
<label>Pad by</label>
|
||||
|
||||
<div>
|
||||
<input id="pad_by_repeat_symbol" value="repeat_symbol" name="pad_by" v-model="toolData.padBy" v-on:change="result"
|
||||
type="radio"> <label for="pad_by_repeat_symbol">repeat symbol</label><br>
|
||||
<input id="pad_by_string_length" value="string_length" name="pad_by" v-model="toolData.padBy" v-on:change="result"
|
||||
type="radio"> <label for="pad_by_string_length">to string length</label>
|
||||
<input
|
||||
id="pad_by_repeat_symbol"
|
||||
value="repeat_symbol"
|
||||
name="pad_by"
|
||||
v-model="toolData.padBy"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="pad_by_repeat_symbol">repeat symbol</label><br />
|
||||
<input
|
||||
id="pad_by_string_length"
|
||||
value="string_length"
|
||||
name="pad_by"
|
||||
v-model="toolData.padBy"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="pad_by_string_length">to string length</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -52,7 +98,7 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
@ -61,9 +107,9 @@ export default {
|
||||
padSymbol: "0",
|
||||
padCount: 3,
|
||||
padPos: "begin",
|
||||
padBy: "string_length"
|
||||
padBy: "string_length",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -81,7 +127,9 @@ export default {
|
||||
return line + padding;
|
||||
}
|
||||
} else {
|
||||
const padding = this.toolData.padSymbol.repeat(this.toolData.padCount);
|
||||
const padding = this.toolData.padSymbol.repeat(
|
||||
this.toolData.padCount
|
||||
);
|
||||
|
||||
if (this.toolData.padPos === "begin") {
|
||||
return padding + line;
|
||||
@ -92,11 +140,9 @@ export default {
|
||||
});
|
||||
|
||||
this.toolResult = paddedLines.join("\n");
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,17 +1,21 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str remove duplicate lines</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<textarea id="data" v-model="toolData.data" v-on:keyup="result"></textarea>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -24,24 +28,25 @@ function arrayUnique(value, index, array) {
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: ""
|
||||
data: "",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
this.toolResult = this.toolData.data.split("\n").filter(arrayUnique).join("\n");
|
||||
}
|
||||
}
|
||||
this.toolResult = this.toolData.data
|
||||
.split("\n")
|
||||
.filter(arrayUnique)
|
||||
.join("\n");
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str sort lines</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -11,25 +11,64 @@
|
||||
<label>Mode</label>
|
||||
|
||||
<div>
|
||||
<input id="sort_mode_alpha" value="alpha" name="sort_mode" v-model="toolData.sortMode" v-on:change="result"
|
||||
type="radio"> <label for="sort_mode_alpha">alphabetically</label><br>
|
||||
<input id="sort_mode_numerically" value="numerically" name="sort_mode" v-model="toolData.sortMode"
|
||||
v-on:change="result" type="radio"> <label for="sort_mode_numerically">numerically</label><br>
|
||||
<input id="sort_mode_by_length" value="by_length" name="sort_mode" v-model="toolData.sortMode"
|
||||
v-on:change="result" type="radio"> <label for="sort_mode_by_length">by length</label><br>
|
||||
<input id="sort_mode_random" value="random" name="sort_mode" v-model="toolData.sortMode" v-on:change="result"
|
||||
type="radio"> <label for="sort_mode_random">random</label><br>
|
||||
<br>
|
||||
<input id="sort_mode_reverse" value="random" name="sort_mode_reverse" v-model="toolData.sortModeReverse"
|
||||
v-on:change="result" type="checkbox"> <label for="sort_mode_reverse">reverse</label>
|
||||
<input
|
||||
id="sort_mode_alpha"
|
||||
value="alpha"
|
||||
name="sort_mode"
|
||||
v-model="toolData.sortMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="sort_mode_alpha">alphabetically</label><br />
|
||||
<input
|
||||
id="sort_mode_numerically"
|
||||
value="numerically"
|
||||
name="sort_mode"
|
||||
v-model="toolData.sortMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="sort_mode_numerically">numerically</label><br />
|
||||
<input
|
||||
id="sort_mode_by_length"
|
||||
value="by_length"
|
||||
name="sort_mode"
|
||||
v-model="toolData.sortMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="sort_mode_by_length">by length</label><br />
|
||||
<input
|
||||
id="sort_mode_random"
|
||||
value="random"
|
||||
name="sort_mode"
|
||||
v-model="toolData.sortMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="sort_mode_random">random</label><br />
|
||||
<br />
|
||||
<input
|
||||
id="sort_mode_reverse"
|
||||
value="random"
|
||||
name="sort_mode_reverse"
|
||||
v-model="toolData.sortModeReverse"
|
||||
v-on:change="result"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="sort_mode_reverse">reverse</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -38,16 +77,16 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
sortMode: "alpha",
|
||||
sortModeReverse: false
|
||||
sortModeReverse: false,
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -57,25 +96,25 @@ export default {
|
||||
|
||||
switch (this.toolData.sortMode) {
|
||||
case "alpha":
|
||||
result = data.sort(function(a, b) {
|
||||
result = data.sort(function (a, b) {
|
||||
return a.localeCompare(b);
|
||||
});
|
||||
break;
|
||||
case "numerically":
|
||||
const collator = new Intl.Collator([], { numeric: true });
|
||||
|
||||
result = data.sort(function(a, b) {
|
||||
result = data.sort(function (a, b) {
|
||||
return collator.compare(a, b);
|
||||
});
|
||||
break;
|
||||
case "by_length":
|
||||
result = data.sort(function(a, b) {
|
||||
result = data.sort(function (a, b) {
|
||||
return a.length - b.length;
|
||||
});
|
||||
break;
|
||||
case "random":
|
||||
result = data.sort(function(a, b) {
|
||||
return Math.random() - .5;
|
||||
result = data.sort(function (a, b) {
|
||||
return Math.random() - 0.5;
|
||||
});
|
||||
break;
|
||||
}
|
||||
@ -85,11 +124,9 @@ export default {
|
||||
}
|
||||
|
||||
this.toolResult = result.join("\n");
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Str to lower/upper</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -11,18 +11,36 @@
|
||||
<label>Mode</label>
|
||||
|
||||
<div>
|
||||
<input id="case_mode_lower" value="lower" name="case_mode" v-model="toolData.caseMode" v-on:change="result"
|
||||
type="radio"> <label for="case_mode_lower">to lower</label><br>
|
||||
<input id="case_mode_upper" value="upper" name="case_mode" v-model="toolData.caseMode" v-on:change="result"
|
||||
type="radio"> <label for="case_mode_upper">to upper</label>
|
||||
<input
|
||||
id="case_mode_lower"
|
||||
value="lower"
|
||||
name="case_mode"
|
||||
v-model="toolData.caseMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="case_mode_lower">to lower</label><br />
|
||||
<input
|
||||
id="case_mode_upper"
|
||||
value="upper"
|
||||
name="case_mode"
|
||||
v-model="toolData.caseMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="case_mode_upper">to upper</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -31,15 +49,15 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
caseMode: "upper"
|
||||
caseMode: "upper",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -52,11 +70,9 @@ export default {
|
||||
this.toolResult = this.toolData.data.toLocaleUpperCase();
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">URL encode/decode</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
@ -11,18 +11,36 @@
|
||||
<label>Mode</label>
|
||||
|
||||
<div>
|
||||
<input id="encode_mode_encode" value="encode" name="encode_mode" v-model="toolData.encodeMode"
|
||||
v-on:change="result" type="radio"> <label for="encode_mode_encode">encode</label><br>
|
||||
<input id="encode_mode_decode" value="decode" name="encode_mode" v-model="toolData.encodeMode"
|
||||
v-on:change="result" type="radio"> <label for="encode_mode_decode">decode</label>
|
||||
<input
|
||||
id="encode_mode_encode"
|
||||
value="encode"
|
||||
name="encode_mode"
|
||||
v-model="toolData.encodeMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="encode_mode_encode">encode</label><br />
|
||||
<input
|
||||
id="encode_mode_decode"
|
||||
value="decode"
|
||||
name="encode_mode"
|
||||
v-model="toolData.encodeMode"
|
||||
v-on:change="result"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="encode_mode_decode">decode</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -31,32 +49,36 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
encodeMode: "decode"
|
||||
encodeMode: "decode",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
switch (this.toolData.encodeMode) {
|
||||
case "encode":
|
||||
this.toolResult = this.toolData.data.split("\n").map(encodeURIComponent).join("\n");
|
||||
this.toolResult = this.toolData.data
|
||||
.split("\n")
|
||||
.map(encodeURIComponent)
|
||||
.join("\n");
|
||||
break;
|
||||
case "decode":
|
||||
this.toolResult = this.toolData.data.split("\n").map(decodeURIComponent).join("\n");
|
||||
this.toolResult = this.toolData.data
|
||||
.split("\n")
|
||||
.map(decodeURIComponent)
|
||||
.join("\n");
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,22 +1,31 @@
|
||||
<template>
|
||||
<h2 class="tool-title">URL query viewer</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<input id="data" v-model="toolData.data" v-on:keyup="result" type="url">
|
||||
<input id="data" v-model="toolData.data" v-on:keyup="result" type="url" />
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<table>
|
||||
<tbody>
|
||||
<tr v-for="row in toolData.queryParts">
|
||||
<td><input :value="row.key" style="width: 100%;" type="text" readonly></td>
|
||||
<td><input :value="row.value" style="width: 100%;" type="text" readonly></td>
|
||||
</tr>
|
||||
<tr v-for="row in toolData.queryParts">
|
||||
<td>
|
||||
<input :value="row.key" style="width: 100%" type="text" readonly />
|
||||
</td>
|
||||
<td>
|
||||
<input
|
||||
:value="row.value"
|
||||
style="width: 100%"
|
||||
type="text"
|
||||
readonly
|
||||
/>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
<p v-if="!toolData.queryParts.length">invalid url</p>
|
||||
@ -41,9 +50,9 @@ export default {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
queryParts: []
|
||||
queryParts: [],
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -63,15 +72,14 @@ export default {
|
||||
for (let [key, value] of url.searchParams.entries()) {
|
||||
queryParams.push({
|
||||
key: key,
|
||||
value: value
|
||||
value: value,
|
||||
});
|
||||
}
|
||||
this.toolData.queryParts = queryParams.sort((a, b) => {
|
||||
return a.key.localeCompare(b.key);
|
||||
});
|
||||
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
@ -83,4 +91,4 @@ table tr:nth-child(even) input {
|
||||
table tr:nth-child(odd) input {
|
||||
background-color: #eee;
|
||||
}
|
||||
</style>
|
||||
</style>
|
||||
|
@ -1,17 +1,28 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Explain crontab</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="data">Data</label>
|
||||
<input id="data" class="input" v-model="toolData.data" v-on:keyup="result" placeholder="* * * * *" type="text">
|
||||
<input
|
||||
id="data"
|
||||
class="input"
|
||||
v-model="toolData.data"
|
||||
v-on:keyup="result"
|
||||
placeholder="* * * * *"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="json" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="json"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -21,20 +32,20 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "* * * * *"
|
||||
data: "* * * * *",
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
this.toolResult = cronstrue.toString(this.toolData.data, {
|
||||
use24HourTimeFormat: true,
|
||||
verbose: true
|
||||
verbose: true,
|
||||
});
|
||||
},
|
||||
methods: {
|
||||
@ -42,16 +53,14 @@ export default {
|
||||
try {
|
||||
this.toolResult = cronstrue.toString(this.toolData.data, {
|
||||
use24HourTimeFormat: true,
|
||||
verbose: true
|
||||
verbose: true,
|
||||
});
|
||||
} catch (e) {
|
||||
this.toolResult = "invalid syntax";
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,6 +1,6 @@
|
||||
<template>
|
||||
<h2 class="tool-title">File base64 encode/decode</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group" v-if="toolData.base64Mode === 'encode'">
|
||||
<label for="data">Data</label>
|
||||
@ -15,8 +15,14 @@
|
||||
<label>Strip mime/type</label>
|
||||
|
||||
<div>
|
||||
<input id="strip_mime" name="strip_mime" v-model="toolData.stripMime" v-on:change="encode" type="checkbox"> <label
|
||||
for="strip_mime">strip mime</label>
|
||||
<input
|
||||
id="strip_mime"
|
||||
name="strip_mime"
|
||||
v-model="toolData.stripMime"
|
||||
v-on:change="encode"
|
||||
type="checkbox"
|
||||
/>
|
||||
<label for="strip_mime">strip mime</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
@ -24,18 +30,34 @@
|
||||
<label>Mode</label>
|
||||
|
||||
<div>
|
||||
<input id="base64_mode_encode" value="encode" name="base64_mode" v-model="toolData.base64Mode" type="radio">
|
||||
<label for="base64_mode_encode">file -> base64</label><br>
|
||||
<input id="base64_mode_decode" value="decode" name="base64_mode" v-model="toolData.base64Mode" type="radio">
|
||||
<input
|
||||
id="base64_mode_encode"
|
||||
value="encode"
|
||||
name="base64_mode"
|
||||
v-model="toolData.base64Mode"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="base64_mode_encode">file -> base64</label><br />
|
||||
<input
|
||||
id="base64_mode_decode"
|
||||
value="decode"
|
||||
name="base64_mode"
|
||||
v-model="toolData.base64Mode"
|
||||
type="radio"
|
||||
/>
|
||||
<label for="base64_mode_decode">base64 -> file</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group" v-if="toolData.base64Mode === 'encode'">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
<div class="input-group" v-if="toolData.base64Mode === 'decode'">
|
||||
<label for="result">Result</label>
|
||||
@ -48,16 +70,16 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
toolData: {
|
||||
data: "",
|
||||
base64Mode: "encode",
|
||||
stripMime: true
|
||||
stripMime: true,
|
||||
},
|
||||
toolResult: ""
|
||||
toolResult: "",
|
||||
};
|
||||
},
|
||||
methods: {
|
||||
@ -78,7 +100,9 @@ export default {
|
||||
}
|
||||
},
|
||||
decode() {
|
||||
const byteCharacters = atob(this.toolData.data.split(",")[1] ?? this.toolData.data);
|
||||
const byteCharacters = atob(
|
||||
this.toolData.data.split(",")[1] ?? this.toolData.data
|
||||
);
|
||||
const byteNumbers = new Array(byteCharacters.length);
|
||||
|
||||
// create file
|
||||
@ -94,11 +118,9 @@ export default {
|
||||
link.href = URL.createObjectURL(blob);
|
||||
link.download = "file";
|
||||
link.click();
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,15 +1,27 @@
|
||||
<template>
|
||||
<h2 class="tool-title">.htaccess User Entry Generator</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="username">Username</label>
|
||||
<input id="username" class="input" v-model="toolData.username" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="username"
|
||||
class="input"
|
||||
v-model="toolData.username"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="password">Password</label>
|
||||
<input id="password" class="input" v-model="toolData.password" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="password"
|
||||
class="input"
|
||||
v-model="toolData.password"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
@ -24,23 +36,27 @@
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="text" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="text"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
import bcrypt from 'bcryptjs';
|
||||
import md5 from 'md5';
|
||||
import sha1 from 'sha1';
|
||||
import bcrypt from "bcryptjs";
|
||||
import md5 from "md5";
|
||||
import sha1 from "sha1";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
@ -58,30 +74,28 @@ export default {
|
||||
methods: {
|
||||
result() {
|
||||
if (!(this.toolData.username && this.toolData.password)) {
|
||||
this.toolResult = '';
|
||||
this.toolResult = "";
|
||||
return;
|
||||
}
|
||||
|
||||
let encryptedPassword = '';
|
||||
let encryptedPassword = "";
|
||||
|
||||
switch (this.toolData.algorithm) {
|
||||
case 'MD5':
|
||||
case "MD5":
|
||||
encryptedPassword = md5(this.toolData.password);
|
||||
break;
|
||||
case 'SHA1':
|
||||
case "SHA1":
|
||||
encryptedPassword = sha1(this.toolData.password);
|
||||
break;
|
||||
case 'BCRYPT':
|
||||
case "BCRYPT":
|
||||
encryptedPassword = bcrypt.hashSync(this.toolData.password, 10); // 10 salt rounds
|
||||
break;
|
||||
}
|
||||
|
||||
this.toolResult = `${this.toolData.username}:${encryptedPassword}`;
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,39 +1,73 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Sed Generator</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="search">Search</label>
|
||||
<input id="search" class="input" v-model="toolData.search" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="search"
|
||||
class="input"
|
||||
v-model="toolData.search"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="replace">Replace</label>
|
||||
<input id="replace" class="input" v-model="toolData.replace" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="replace"
|
||||
class="input"
|
||||
v-model="toolData.replace"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="delimiter">Delimiter</label>
|
||||
<input id="delimiter" class="input" maxlength="1" v-model="toolData.delimiter" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="delimiter"
|
||||
class="input"
|
||||
maxlength="1"
|
||||
v-model="toolData.delimiter"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<label for="filename">Filename</label>
|
||||
<input id="filename" class="input" v-model="toolData.filename" v-on:keyup="result" type="text">
|
||||
<input
|
||||
id="filename"
|
||||
class="input"
|
||||
v-model="toolData.filename"
|
||||
v-on:keyup="result"
|
||||
type="text"
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div class="input-group">
|
||||
<div>
|
||||
<input id="do_replace" type="checkbox" v-model="toolData.doReplace" v-on:change="result">
|
||||
<input
|
||||
id="do_replace"
|
||||
type="checkbox"
|
||||
v-model="toolData.doReplace"
|
||||
v-on:change="result"
|
||||
/>
|
||||
<label for="do_replace"> replace in file</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="result">Result</label>
|
||||
<MonacoEditor name="result" language="bash" :value="toolResult"></MonacoEditor>
|
||||
<MonacoEditor
|
||||
name="result"
|
||||
language="bash"
|
||||
:value="toolResult"
|
||||
></MonacoEditor>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@ -42,7 +76,7 @@ import MonacoEditor from "@/components/MonacoEditor.vue";
|
||||
|
||||
export default {
|
||||
components: {
|
||||
MonacoEditor
|
||||
MonacoEditor,
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
@ -61,9 +95,15 @@ export default {
|
||||
},
|
||||
methods: {
|
||||
result() {
|
||||
if (!(this.toolData.search && this.toolData.replace && this.toolData.delimiter)) {
|
||||
this.toolResult = '';
|
||||
return
|
||||
if (
|
||||
!(
|
||||
this.toolData.search &&
|
||||
this.toolData.replace &&
|
||||
this.toolData.delimiter
|
||||
)
|
||||
) {
|
||||
this.toolResult = "";
|
||||
return;
|
||||
}
|
||||
|
||||
let command = `sed`;
|
||||
@ -72,8 +112,14 @@ export default {
|
||||
}
|
||||
|
||||
// Escape delimiter
|
||||
const search = this.toolData.search.replace(new RegExp(`\\${this.toolData.delimiter}`, 'g'), `\\${this.toolData.delimiter}`);
|
||||
const replace = this.toolData.replace.replace(new RegExp(`\\${this.toolData.delimiter}`, 'g'), `\\${this.toolData.delimiter}`);
|
||||
const search = this.toolData.search.replace(
|
||||
new RegExp(`\\${this.toolData.delimiter}`, "g"),
|
||||
`\\${this.toolData.delimiter}`
|
||||
);
|
||||
const replace = this.toolData.replace.replace(
|
||||
new RegExp(`\\${this.toolData.delimiter}`, "g"),
|
||||
`\\${this.toolData.delimiter}`
|
||||
);
|
||||
|
||||
command += ` 's${this.toolData.delimiter}${search}${this.toolData.delimiter}${replace}${this.toolData.delimiter}g' ${this.toolData.filename}`;
|
||||
this.toolResult = command;
|
||||
@ -82,6 +128,4 @@ export default {
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss">
|
||||
|
||||
</style>
|
||||
<style lang="scss"></style>
|
||||
|
@ -1,33 +1,38 @@
|
||||
<template>
|
||||
<h2 class="tool-title">Unix timestamp</h2>
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<table>
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Format</th>
|
||||
<th>Date & Time</th>
|
||||
<th></th>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Format</th>
|
||||
<th>Date & Time</th>
|
||||
<th></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr v-for="(value, format) in currentTimestampFormatted" :key="format">
|
||||
<td>{{ format }}</td>
|
||||
<td>{{ value }}</td>
|
||||
<td>
|
||||
<button @click="copyTimestamp(value)">📋</button>
|
||||
</td>
|
||||
</tr>
|
||||
<tr v-for="(value, format) in currentTimestampFormatted" :key="format">
|
||||
<td>{{ format }}</td>
|
||||
<td>{{ value }}</td>
|
||||
<td>
|
||||
<button @click="copyTimestamp(value)">📋</button>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
|
||||
<hr class="mt-5 mb-5">
|
||||
<hr class="mt-5 mb-5" />
|
||||
|
||||
<div class="input-group">
|
||||
<label for="inputTimestamp">Enter timestamp</label>
|
||||
|
||||
<div>
|
||||
<input id="inputTimestamp" type="number" v-model="toolData.inputTimestamp" :placeholder="currentTimestamp">
|
||||
<input
|
||||
id="inputTimestamp"
|
||||
type="number"
|
||||
v-model="toolData.inputTimestamp"
|
||||
:placeholder="currentTimestamp"
|
||||
/>
|
||||
<button @click="convertFromUnix">Convert ↓</button>
|
||||
</div>
|
||||
</div>
|
||||
@ -38,28 +43,28 @@
|
||||
<div>
|
||||
<table class="table-date-and-time">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Year</th>
|
||||
<th>Month</th>
|
||||
<th>Day</th>
|
||||
<th>Hour (24)</th>
|
||||
<th>Minute</th>
|
||||
<th>Second</th>
|
||||
<th></th>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Year</th>
|
||||
<th>Month</th>
|
||||
<th>Day</th>
|
||||
<th>Hour (24)</th>
|
||||
<th>Minute</th>
|
||||
<th>Second</th>
|
||||
<th></th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr>
|
||||
<td><input type="number" v-model="toolData.inputYear"></td>
|
||||
<td><input type="number" v-model="toolData.inputMonth"></td>
|
||||
<td><input type="number" v-model="toolData.inputDay"></td>
|
||||
<td><input type="number" v-model="toolData.inputHour"></td>
|
||||
<td><input type="number" v-model="toolData.inputMinute"></td>
|
||||
<td><input type="number" v-model="toolData.inputSecond"></td>
|
||||
<td>
|
||||
<button @click="convertToUnix">Convert ↑</button>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td><input type="number" v-model="toolData.inputYear" /></td>
|
||||
<td><input type="number" v-model="toolData.inputMonth" /></td>
|
||||
<td><input type="number" v-model="toolData.inputDay" /></td>
|
||||
<td><input type="number" v-model="toolData.inputHour" /></td>
|
||||
<td><input type="number" v-model="toolData.inputMinute" /></td>
|
||||
<td><input type="number" v-model="toolData.inputSecond" /></td>
|
||||
<td>
|
||||
<button @click="convertToUnix">Convert ↑</button>
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
@ -79,10 +84,10 @@ export default {
|
||||
inputDay: moment().day(),
|
||||
inputHour: moment().hour(),
|
||||
inputMinute: moment().minute(),
|
||||
inputSecond: moment().second()
|
||||
inputSecond: moment().second(),
|
||||
},
|
||||
currentTimestamp: 0,
|
||||
currentTimestampFormatted: {}
|
||||
currentTimestampFormatted: {},
|
||||
};
|
||||
},
|
||||
mounted() {
|
||||
@ -90,7 +95,7 @@ export default {
|
||||
this.currentTimestamp = moment().unix();
|
||||
|
||||
this.currentTimestampFormatted = {
|
||||
"UTC": moment().utc().format("YYYY-MM-DDTHH:mm:ss[Z]"),
|
||||
UTC: moment().utc().format("YYYY-MM-DDTHH:mm:ss[Z]"),
|
||||
// 'UTC (local)': moment().format('YYYY-MM-DDTHH:mm:ssZZ'),
|
||||
// 'ISO 8601': moment().toISOString(),
|
||||
// 'RFC 2822': moment().format('ddd, DD MMM YYYY HH:mm:ss ZZ'),
|
||||
@ -103,8 +108,10 @@ export default {
|
||||
// 'RSS': moment().format('ddd, DD MMM YYYY HH:mm:ss ZZ'),
|
||||
"Unix Epoch": moment().unix(),
|
||||
"YYYY-DD-MM": moment().format("YYYY/MM/DD"),
|
||||
"SQL DATETIME": moment().toISOString().replaceAll(/[ZT]|(\.[0-9]{3})/g, " "),
|
||||
"YYYY/DD/MM HH:MM:SS AM/PM": moment().format("YYYY/MM/DD HH:mm:ss A")
|
||||
"SQL DATETIME": moment()
|
||||
.toISOString()
|
||||
.replaceAll(/[ZT]|(\.[0-9]{3})/g, " "),
|
||||
"YYYY/DD/MM HH:MM:SS AM/PM": moment().format("YYYY/MM/DD HH:mm:ss A"),
|
||||
// 'DD.MM.YYYY HH:MM:SS': moment().format('DD.MM.YYYY HH:mm:ss'),
|
||||
};
|
||||
};
|
||||
@ -114,7 +121,9 @@ export default {
|
||||
},
|
||||
methods: {
|
||||
convertFromUnix() {
|
||||
let timestamp = this.toolData.inputTimestamp.length ? this.toolData.inputTimestamp : this.currentTimestamp;
|
||||
let timestamp = this.toolData.inputTimestamp.length
|
||||
? this.toolData.inputTimestamp
|
||||
: this.currentTimestamp;
|
||||
|
||||
const date = moment.unix(timestamp);
|
||||
|
||||
@ -124,7 +133,6 @@ export default {
|
||||
this.toolData.inputHour = date.hour();
|
||||
this.toolData.inputMinute = date.minute();
|
||||
this.toolData.inputSecond = date.second();
|
||||
|
||||
},
|
||||
convertToUnix() {
|
||||
const date = moment({
|
||||
@ -133,21 +141,22 @@ export default {
|
||||
day: this.toolData.inputDay,
|
||||
hour: this.toolData.inputHour,
|
||||
minute: this.toolData.inputMinute,
|
||||
second: this.toolData.inputSecond
|
||||
second: this.toolData.inputSecond,
|
||||
});
|
||||
|
||||
this.toolData.inputTimestamp = date.unix();
|
||||
},
|
||||
copyTimestamp(text) {
|
||||
navigator.clipboard.writeText(text)
|
||||
navigator.clipboard
|
||||
.writeText(text)
|
||||
.then(() => {
|
||||
// console.log('Timestamp copied to clipboard!');
|
||||
})
|
||||
.catch(err => {
|
||||
.catch((err) => {
|
||||
// console.error('Failed to copy: ', err);
|
||||
});
|
||||
}
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
@ -159,4 +168,4 @@ export default {
|
||||
.table-date-and-time td input {
|
||||
width: 80px;
|
||||
}
|
||||
</style>
|
||||
</style>
|
||||
|
@ -1,17 +1,17 @@
|
||||
module.exports = {
|
||||
content: [
|
||||
"./index.html",
|
||||
"./src/**/*.{vue,js,ts,jsx,tsx}",
|
||||
// "./node_modules/flowbite/**/*.js",
|
||||
],
|
||||
theme: {
|
||||
extend: {
|
||||
zIndex: {
|
||||
'500': '500',
|
||||
}
|
||||
},
|
||||
},
|
||||
plugins: [
|
||||
// require('flowbite/plugin')
|
||||
],
|
||||
}
|
||||
content: [
|
||||
"./index.html",
|
||||
"./src/**/*.{vue,js,ts,jsx,tsx}",
|
||||
// "./node_modules/flowbite/**/*.js",
|
||||
],
|
||||
theme: {
|
||||
extend: {
|
||||
zIndex: {
|
||||
500: "500",
|
||||
},
|
||||
},
|
||||
},
|
||||
plugins: [
|
||||
// require('flowbite/plugin')
|
||||
],
|
||||
};
|
||||
|
@ -1,29 +1,29 @@
|
||||
import {fileURLToPath, URL} from 'url'
|
||||
import { fileURLToPath, URL } from "url";
|
||||
|
||||
import {defineConfig} from 'vite'
|
||||
import vue from '@vitejs/plugin-vue'
|
||||
import { defineConfig } from "vite";
|
||||
import vue from "@vitejs/plugin-vue";
|
||||
|
||||
function pathTo(path) {
|
||||
return fileURLToPath(new URL(path, import.meta.url));
|
||||
return fileURLToPath(new URL(path, import.meta.url));
|
||||
}
|
||||
|
||||
// https://vitejs.dev/config/
|
||||
export default defineConfig({
|
||||
plugins: [vue()],
|
||||
resolve: {
|
||||
alias: {
|
||||
'@': pathTo('./src', import.meta.url),
|
||||
'@node_modules': pathTo('./node_modules'),
|
||||
}
|
||||
},
|
||||
server: {
|
||||
host: true,
|
||||
port: 3000,
|
||||
cors: false,
|
||||
},
|
||||
preview: {
|
||||
host: true,
|
||||
port: 3000,
|
||||
cors: false,
|
||||
}
|
||||
})
|
||||
plugins: [vue()],
|
||||
resolve: {
|
||||
alias: {
|
||||
"@": pathTo("./src", import.meta.url),
|
||||
"@node_modules": pathTo("./node_modules"),
|
||||
},
|
||||
},
|
||||
server: {
|
||||
host: true,
|
||||
port: 3000,
|
||||
cors: false,
|
||||
},
|
||||
preview: {
|
||||
host: true,
|
||||
port: 3000,
|
||||
cors: false,
|
||||
},
|
||||
});
|
||||
|
Reference in New Issue
Block a user