diff --git a/.env_template b/.env_template new file mode 100644 index 0000000..635b15d --- /dev/null +++ b/.env_template @@ -0,0 +1,10 @@ +FIREBASE_API_KEY= +FIREBASE_AUTH_DOMAIN= +FIREBASE_PROJECT_ID= +FIREBASE_STORAGE_BUCKET= +FIREBASE_MESSAGING_SENDER_ID= +FIREBASE_APP_ID= +FIREBASE_MEASUREMENT_ID= +CLIENT_ID= +EXTENSION_KEY= +BACKEND_URL= \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..22df66f --- /dev/null +++ b/.gitignore @@ -0,0 +1,16 @@ +npm-debug.log +coverage/ +node_modules/ +package-lock.json +/vendor/ +/*.zip +dist/ +.env + +server/.env +server/firebaseServiceAccountKey.json +server/.env +server/data +**/__pycache__/** +server/.venv +server/model/* \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md new file mode 100644 index 0000000..edf146b --- /dev/null +++ b/CHANGELOG.md @@ -0,0 +1,2 @@ + +# Change Log \ No newline at end of file diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..edbeb99 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2023 Ian Lee + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md index edf5490..dfe19eb 100644 --- a/README.md +++ b/README.md @@ -1,11 +1,13 @@ -# Manga Reader Browser Extension +# Veebee Browser Extension -**Install: [Chrome](https://chrome.google.com/webstore/detail/manga-reader/eabnmbpmoencafnpbobahdeamaljhoef)** +Uses Google Vision, DeepL, GPT, and [manga-ocr](https://github.com/kha-white/manga-ocr) to help you with translating and/or learning new languages while browing the internet in a Chrome Browser. One great use case is for reading manga but you can use it on most content on the web. + +![](assets/example.png) -# Origins +**Install: [Chrome](https://chrome.google.com/webstore/detail/manga-reader/eabnmbpmoencafnpbobahdeamaljhoef)** -Note that the extension won't work on certain origins: +Note that the extension won't work on certain origins, such as: - chrome origins like: `chrome://` and `chrome-extension://` - the official chrome web store: `https://chrome.google.com/webstore/category/extensions` @@ -21,17 +23,30 @@ Note that the extension won't work on certain origins: ## Build Backend Locally 1. Fill in the `server/.env_template` and rename it to `server/.env` -2. Set up GCP project with the correct (Optional if you comment out the GCP related stuff) +2. Set up GCP project (Optional if you comment out the GCP related stuff) 3. Create your `server/firebaseServiceAccountKey.json` (Optional if you comment out the authentication stuff) 4. Set up Firestore (Optional if you comment out the Firestore related code) 5. `pip install -r requirements.txt` 6. `python3 main.py` +# Contact + +Please send any feedback or inquiries to lyrian1029@gmail.com + +This is an early Proof of Concept. If users like it, I am willing to invest more time to add various features, such as translating an entire page at a time, adding side panel to allow more space for translation details, and chat mode with GPT. Another obvious one is to add more languages. Any other feedback is very welcome! + + +# Acknowledgments + +Learned a lot about Chrome extensions from +https://github.com/RasikaWarade/chrome-extension-mv3-firebase-boilerplate +https://github.com/simov/screenshot-capture + # License The MIT License (MIT) -Copyright (c) 2023-present Ian Lee (https://github.com/ianbbqzy/manga-reader) +Copyright (c) 2023-present Ian Lee (https://github.com/ianbbqzy/veebee) Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/assets/example.png b/assets/example.png new file mode 100644 index 0000000..c67c324 Binary files /dev/null and b/assets/example.png differ diff --git a/background/background.js b/background/background.js new file mode 100644 index 0000000..e0f0f2a --- /dev/null +++ b/background/background.js @@ -0,0 +1,156 @@ +// for stuff that doesn't involve the DOM of the web page the user is viewing + +// Set default config values +chrome.storage.sync.get((config) => { + if (!config.api) { + chrome.storage.sync.set({api: 'deepl'}) + } + + if (!config.source_lang) { + chrome.storage.sync.set({source_lang: 'Japanese'}) + } + + if (!config.target_lang) { + chrome.storage.sync.set({target_lang: 'English'}) + } + + if (config.icon === undefined) { + config.icon = false + chrome.storage.sync.set({icon: false}) + } + + chrome.action.setIcon({ + path: [16, 19, 38, 48, 128].reduce((all, size) => ( + color = config.icon ? 'light' : 'dark', + all[size] = `/icons/${color}/${size}x${size}.png`, + all + ), {}) + }) +}) + +// This is triggered when extension icon is clicked. This is the main entry point +// for screenshot capture. +// It injects the content script into the active tab. +chrome.action.onClicked.addListener((tab) => { + pingContentScript(tab, 'initCrop'); +}) + +// take-screenshot is received when keyboard shortcut is triggered, as defined in manifest.json +// This is another entry point for screenshot capture. +chrome.commands.onCommand.addListener((command) => { + if (command === 'take-screenshot') { + chrome.tabs.query({active: true, currentWindow: true}, (tabs) => { + pingContentScript(tabs[0], 'initCrop'); + }) + } +}) + +// capture request is received when the user cropping by the user is done. +// active rquest is received when +chrome.runtime.onMessage.addListener((req, sender, res) => { + if (req.message === 'capture') { + chrome.tabs.query({active: true, currentWindow: true}, (tabs) => { + chrome.tabs.captureVisibleTab(tabs[0].windowId, (image) => { + res({message: 'image', image: image}) + }) + }) + } + else if (req.message === 'active') { + // Change the extension icon and title based on whether the user is cropping + if (req.active) { + chrome.storage.sync.get(() => { + chrome.action.setTitle({tabId: sender.tab.id, title: 'Crop'}) + chrome.action.setBadgeText({tabId: sender.tab.id, text: '◩'}) + }) + } + else { + chrome.action.setTitle({tabId: sender.tab.id, title: 'Crop Initialized'}) + chrome.action.setBadgeText({tabId: sender.tab.id, text: ''}) + } + } + return true +}) + +// Create context menu option +chrome.runtime.onInstalled.addListener(() => { + chrome.contextMenus.create({ + id: 'translate-menu', + // chrome replaces %s with whatever is highlighted + title: 'Translate %s', + // selects what's hightlighted by the cursor? + contexts: ['selection'] + }); +}); + +// Handle when context menu is clicked +chrome.contextMenus.onClicked.addListener((info, tab) => { + console.log(info.menuItemId) + if (info.menuItemId === "translate-menu") { + // Let content script know that a text translation is initialized. + // This has to be done in background script in case the content + // script has not been initialized. + pingContentScript(tab, 'initTextTranslation'); + + // continue with the translation process. + chrome.storage.sync.get((config) => { + if (!config.idToken) { + pingContentScript(tab, "Please login first. Right click on the extension icon and click on options.") + } else { + callTranslateWithText(info.selectionText, config.source_lang, config.target_lang, config.api, config.idToken) + .then(response => { + pingContentScript(tab, response) + }) + .catch(error => { + console.error(`error: ${error.message}`); + pingContentScript(tab, `error: ${error.message}`) + }); + } + }) + } +}); + +async function callTranslateWithText(text, source_lang, target_lang, api, idToken) { + const url = "__BACKEND_URL__"; + const headers = new Headers(); + headers.append('Authorization', `Bearer ${idToken}`); + headers.append('Content-Type', `application/json`); + + try { + const resp = await fetch(url + '/translate-text?api=' + api + '&source_lang=' + source_lang + '&target_lang=' + target_lang, { + method: 'POST', + headers: headers, + body: JSON.stringify({ + 'text': text + }) + }).then(res => res.json()) + if (resp.error) { + return `Translation: ${resp.error}`; + } + return `Translation: ${resp.translation}`; + } catch (err) { + return `Translation: ${err.message}`; + } +} + +// Sends a message to the content script +// If it doesn't receive a response within a specific timeout, it +// determines that content script is not initialized and it will initialize it. +// After another small timeout, it'll try to send the message again. +function pingContentScript(tab, message) { + chrome.tabs.sendMessage(tab.id, {message: message}, (res) => { + if (res) { + // if response is received before the timeout is triggered + // clears the timeout call + clearTimeout(timeout) + } + }) + + var timeout = setTimeout(() => { + chrome.scripting.insertCSS({files: ['css/content.css'], target: {tabId: tab.id}}) + chrome.scripting.executeScript({files: ['content.js'], target: {tabId: tab.id}}) + + setTimeout(() => { + chrome.tabs.sendMessage(tab.id, {message: message}) + }, 100) + }, 100) +} diff --git a/content/content.js b/content/content.js new file mode 100644 index 0000000..8af8599 --- /dev/null +++ b/content/content.js @@ -0,0 +1,334 @@ +import $ from 'jquery'; +import Jcrop from 'jquery-jcrop'; +import 'jquery-jcrop/css/jquery.Jcrop.min.css'; +import 'dotenv/config' + +let jcrop, selection; + +// Handles messages +// currently we only expect messages from the background script. +chrome.runtime.onMessage.addListener((req, sender, res) => { + // Sends a quick response to background script, which will + // use the response to prevent re-injection of the content script. + res({}) + + console.log(req.message); + if (req.message === 'initCrop') { + // If jcrop is not initialized, initialize it. + // TODO: maybe we can initialize this on page load? + if (!jcrop) { + console.log("jcrop not initialized") + // create fake image, then init jcrop, then call overlay() and capture() + image(() => init(() => { + jcropOverlay(true) + capture() + })) + } + else { + // jcrop already initialized. In this case, if there is already a cropping + // session, ends it. If not, starts one. so we call overlay() to toggle + // the active state. + jcropOverlay() + capture() + } + } else if (req.message === "initTextTranslation") { + // translation requested + showTextTranslationDialog("translating") + } else { + // If neither of the above, the message is the result + // of the translation request. It could also be an error message. + showTextTranslationDialog(req.message) + } + return true +}) + +// uses a closure to maintain its state (active). It toggles the +// visibility of the cropping area (jcrop-holder) and sends a message to +// the background script about whether the cropping is active or not. +// +// By passing false and invoking the outer function immediately, the code +// is setting an initial value for active which is retained and can be +// accessed/modified every time you call overlay. This provides a way to +// maintain state (active) between calls to overlay without exposing this +// state to the external world, thus encapsulating the behavior and state. +// +// If call with no argument, it toggles the active state. +// If call with true or false, it sets the active state. +// if call with NULL, it does not change the active state. +const jcropOverlay = ((active) => (state) => { + active = typeof state === 'boolean' ? state : state === null ? active : !active; + $('.jcrop-holder')[active ? 'show' : 'hide'](); + chrome.runtime.sendMessage({message: 'active', active}); +})(false); + +// creates an "invisible" image (pixel.png) for Jcrop to bind to when +// initializing the cropping tool. a workaround to get Jcrop to +// initialize without needing a real image. +const image = (done) => { + const img = new Image(); + img.id = 'fake-image'; + img.src = chrome.runtime.getURL('/icons/pixel.png'); + img.onload = () => { + $('body').append(img); + done(); + }; +}; + +// only invoked after image() has been called +// initializes Jcrop on the "invisible" image, with various event +// handlers for user interactions. +const init = (done) => { + console.log("initing jcrop"); + // Jcrop responsible for setting selection + $('#fake-image').Jcrop({ + bgColor: 'none', + onSelect: (e) => { + selection = e; + capture(); + }, + }, function ready() { + jcrop = this; + + // import jcropGif from 'jquery-jcrop/css/Jcrop.gif' doesn't work. + $('.jcrop-hline, .jcrop-vline').css({ + backgroundImage: `url(${chrome.runtime.getURL('/icons/Jcrop.gif')})` + }); + + done && done(); + }); +}; + +// Process crop area after Jcrop has set the selection. +var capture = () => { + chrome.storage.sync.get((config) => { + // selection is set by Jcrop + if (selection) { + // save the coordinates so that selection can be cleared + const coordinates = {...selection} + jcrop.release() + selection = null + jcropOverlay(false) + + // Send message to background script to capture a screenshot of the + // entire page and responds with the image. + chrome.runtime.sendMessage({message: 'capture'}, (res) => { + if (!config.idToken) { + showTranslationDialog("Please login first. Right click on the extension icon and click on options.", coordinates, "") + } else { + // With the screenshot from the background script, crops the screenshot + // using selection. Then get the translation of the text in the image. + crop(res.image, coordinates, (image) => { + getTranslation(image, coordinates, config.api, config.idToken, config.source_lang, config.target_lang) + }) + showTranslationDialog("translating", coordinates, "") + } + + }) + } + }) +} + +async function callTranslateWithScreenshot(image, source_lang, target_lang, api, idToken) { + const url = process.env.BACKEND_URL; + const headers = new Headers(); + headers.append('Authorization', `Bearer ${idToken}`); + headers.append('Content-Type', `application/json`); + + try { + const resp = await fetch(url + '/translate-img?api=' + api + '&source_lang=' + source_lang + '&target_lang=' + target_lang, { + method: 'POST', + headers: headers, + body: JSON.stringify({ + 'imageDataUrl': image + }) + }).then(res => res.json()) + + if (resp.error) { + return `Translation: ${resp.error}`; + } + return {"translation": `Translation:\n${resp.translation}`, "original": resp.original}; + } catch (err) { + return `Translation: ${err.message}`; + } +} + +var getTranslation = async (image, coordinates, api, idToken, source_lang, target_lang) => { + // Call the function asynchronously without awaiting it + callTranslateWithScreenshot(image, source_lang, target_lang, api, idToken) + .then(response => { + if (response.translation) { + showTranslationDialog(response.translation, coordinates, response.original) + } else { + showTranslationDialog(response, coordinates, "") + } + }) + .catch(error => { + console.error(`Error: ${error.message}`); + showTranslationDialog(`Error: ${error.message}`, coordinates, "") + }); +} + +// if window is resized while cropping, re init jcrop. +window.addEventListener('resize', ((timeout) => () => { + jcrop.destroy() + init(() => jcropOverlay(null)) +})()) + +/* + * Display dialog box with translation when selecting text + * handled differently when selecting a text from an existing overlay + */ +function showTextTranslationDialog(translation) { + // Get selection to know where to position the dialog + const selection = window.getSelection(); + if (!selection) { + console.log("Nothing was selected"); + return; + } + const rect = selection.getRangeAt(0).getBoundingClientRect(); + + // Check if the selection is within an existing overlay + const existingOverlay = document.querySelector("#overlay") || document.querySelector("#testTranslationOverlay"); + if (existingOverlay && isSelectionInsideElement(existingOverlay)) { + const overlayRect = existingOverlay.getBoundingClientRect(); + + console.log("Selection is inside an existing overlay") + showTranslationDialog(translation,{ + x: overlayRect.left, + y: overlayRect.top, + x2: overlayRect.right, + y2: overlayRect.bottom, + }, selection.toString(), "testTranslationOverlay"); + } else { + console.log("Selection is not inside an existing overlay") + // use the same approach as image translation to determine where to spawn the overlay + showTranslationDialog(translation, { + x: rect.left, + y: rect.top, + x2: rect.right, + y2: rect.bottom, + }, selection.toString(), "testTranslationOverlay"); + } +} + +function showTranslationDialog(translation, coordinates, original, overlayID = 'overlay') { + const viewportWidth = Math.max(document.documentElement.clientWidth, window.innerWidth || 0); + const viewportCenterX = (viewportWidth / 2) + window.scrollX; + const rectCenterX = (coordinates.x + coordinates.x2) / 2; + + // Determine the spawn position of the overlay based on rectangle position + const spawnX = rectCenterX <= viewportCenterX + ? coordinates.x2 + window.scrollX + : coordinates.x - 300 + window.scrollX; // Assuming the overlay width is 300px + + // Remove existing overlay if present + const existingOverlay = document.querySelector("#" + overlayID); + if (existingOverlay) existingOverlay.remove(); + + // Create and append the overlay + const overlay = document.createElement('div'); + overlay.id = overlayID; + overlay.style.cssText = ` + background-color: white; + font-size: 16px; + border: 1px solid #cccccc; + width: 300px; + white-space: pre-wrap; + position: absolute; + top: ${coordinates.y + window.scrollY}px; + left: ${spawnX}px; + z-index: 999; + `; + + overlay.innerHTML = ` +

${translation}

+ +
+ + + +
+ `; + document.body.appendChild(overlay); + attachEventListeners(overlayID); +} + +function minimizeOverlay(overlayID) { + const overlay = document.querySelector("#" + overlayID); + overlay.dataset.initialHtml = overlay.innerHTML; + overlay.style.width = "30px"; + overlay.style.height = "30px"; + overlay.innerHTML = ``; + document.querySelector("#overlay-restore-button" + overlayID).addEventListener("click", () => restoreOverlay(overlayID)); +} + +function restoreOverlay(overlayID) { + const overlay = document.querySelector("#" + overlayID); + overlay.style.width = "300px"; + overlay.style.height = "auto"; + overlay.innerHTML = overlay.dataset.initialHtml; + attachEventListeners(overlayID); +} + +function attachEventListeners(overlayID) { + const overlay = document.querySelector("#" + overlayID); + + document.querySelector("#overlay-minimize-button" + overlayID).addEventListener("click", () => minimizeOverlay(overlayID)); + document.querySelector("#overlay-close-button" + overlayID).addEventListener("click", () => overlay.remove()); + + const toggleButton = document.getElementById("toggleButton" + overlayID); + toggleButton.addEventListener("click", function() { + const translationElement = document.getElementById("translation" + overlayID); + const originalElement = document.getElementById("original" + overlayID); + + const isTranslationVisible = translationElement.style.display !== "none"; + translationElement.style.display = isTranslationVisible ? "none" : "block"; + originalElement.style.display = isTranslationVisible ? "block" : "none"; + }); +} + +function crop (image, area, done) { + const dpr = devicePixelRatio + console.log("area") + console.log(area) + var top = area.y * dpr + var left = area.x * dpr + var width = area.w * dpr + var height = area.h * dpr + var w = (dpr !== 1) ? width : area.w + var h = (dpr !== 1) ? height : area.h + + var canvas = null + var template = null + if (!canvas) { + template = document.createElement('template') + canvas = document.createElement('canvas') + document.body.appendChild(template) + template.appendChild(canvas) + } + canvas.width = w + canvas.height = h + + var img = new Image() + img.onload = () => { + var context = canvas.getContext('2d') + context.drawImage(img, + left, top, + width, height, + 0, 0, + w, h + ) + + var cropped = canvas.toDataURL(`image/png`) + done(cropped) + } + img.src = image +} + +function isSelectionInsideElement(element) { + const selection = window.getSelection(); + if (selection.rangeCount === 0) return false; + + const node = selection.anchorNode; + return element.contains(node); +} diff --git a/css/button.css b/css/button.css new file mode 100644 index 0000000..a9946e2 --- /dev/null +++ b/css/button.css @@ -0,0 +1,13 @@ +button { + height: 30px; + width: 30px; + outline: none; + margin: 10px; + border: none; + border-radius: 2px; + } + + button.current { + box-shadow: 0 0 0 2px white, + 0 0 0 4px black; + } \ No newline at end of file diff --git a/css/content.css b/css/content.css new file mode 100644 index 0000000..7e9c3d3 --- /dev/null +++ b/css/content.css @@ -0,0 +1,18 @@ + +img#fake-image, +.jcrop-holder, +.jcrop-holder img, +.jcrop-tracker { + width: 100% !important; height: 100% !important; + max-width: 100% !important; max-height: 100% !important; + min-width: 100% !important; min-height: 100% !important; +} + +img#fake-image { + position: fixed; top: 0; left: 0; z-index: 1; +} + +.jcrop-holder { + position: fixed !important; top: 0 !important; left: 0 !important; + z-index: 2147483647 !important; +} diff --git a/css/icons.css b/css/icons.css new file mode 100644 index 0000000..aaed067 --- /dev/null +++ b/css/icons.css @@ -0,0 +1,55 @@ +@font-face { + font-family: 'fontello'; + src: url('icons.ttf') format('truetype'); + font-weight: normal; + font-style: normal; +} +/* Chrome hack: SVG is rendered more smooth in Windozze. 100% magic, uncomment if you need it. */ +/* Note, that will break hinting! In other OS-es font will be not as sharp as it could be */ +/* +@media screen and (-webkit-min-device-pixel-ratio:0) { + @font-face { + font-family: 'fontello'; + src: url('../font/fontello.svg?19777476#fontello') format('svg'); + } +} +*/ + + [class^="icon-"]:before, [class*=" icon-"]:before { + font-family: "fontello"; + font-style: normal; + font-weight: normal; + speak: none; + + display: inline-block; + text-decoration: inherit; + width: 1em; + margin-right: .2em; + text-align: center; + /* opacity: .8; */ + + /* For safety - reset parent styles, that can break glyph codes*/ + font-variant: normal; + text-transform: none; + + /* fix buttons height, for twitter bootstrap */ + line-height: 1em; + + /* Animation center compensation - margins should be symmetric */ + /* remove if not needed */ + margin-left: .2em; + + /* you can be more comfortable with increased icons size */ + /* font-size: 120%; */ + + /* Font smoothing. That was taken from TWBS */ + -webkit-font-smoothing: antialiased; + -moz-osx-font-smoothing: grayscale; + + /* Uncomment for 3D effect */ + /* text-shadow: 1px 1px 1px rgba(127, 127, 127, 0.3); */ +} + +.icon-github:before { content: '\f09b'; } /* '' */ +.icon-menu:before { content: '\f0c9'; } /* '' */ +.icon-chrome:before { content: '\f268'; } /* '' */ diff --git a/css/icons.ttf b/css/icons.ttf new file mode 100644 index 0000000..2cfd09d Binary files /dev/null and b/css/icons.ttf differ diff --git a/css/main.css b/css/main.css new file mode 100644 index 0000000..e7fdbcb --- /dev/null +++ b/css/main.css @@ -0,0 +1,72 @@ +@import "https://maxcdn.bootstrapcdn.com/font-awesome/4.5.0/css/font-awesome.min.css"; +body { + display: block; + flex-direction: column; + justify-content: center; + background: linear-gradient(#fff, #3c76b8); + font-family: "Lato", sans-serif; + width: 300px; + height: 230px; +} + +h1 { + color: #3c76b8; + text-align: center; + font-size: 40px; + font-weight: 100; + margin: 0 0 30px; + padding-left: 10px; + padding-top: 5px; +} + +.btn { + display: block; + border: 1px solid transparent; + height: 35px; + width: 100%; + border-radius: 5px; + transition: all 0.5s ease; + cursor: pointer; + outline: none; + font-family: "Lato", sans-serif; + font-weight: 600; + margin: 0 0 30px; +} +.btn__color{ + display: flex; + border: 0px transparent; + height: 50px; + border-radius: 5px; + cursor: pointer; + outline: none; +} + +.btn__sign_out { + background-color: #4285F4; + color: #fff; + font-size: 18px; +} + +.btn__sign_out:hover { + background-color: #fff; + color: #4285F4; + border: 1px solid #4285F4; +} + +#sign_out{ + width: 100%; +} + +@keyframes choosing { + 100% { + transform: rotate(0deg) scale(1.5); + box-shadow: 0px 0px 20px 2px white; ; + } +} + +#sign_out:hover { + animation-name: choosing; + animation-direction: normal; + animation-fill-mode: forwards; + animation-duration: 0.4s; +} \ No newline at end of file diff --git a/css/options.css b/css/options.css new file mode 100644 index 0000000..24e9ef8 --- /dev/null +++ b/css/options.css @@ -0,0 +1,160 @@ + +/*sticky footer*/ +html, body { + height: 100%; + padding: 0; + margin: 0; +} +#wrapper { + min-height: 100%; + height: auto !important; + height: 100%; + margin: 0 auto -52px; +} +footer, #footer-push { + height: 20px; + text-align: center; +} + + +/*center content on large screens*/ +.mdc-toolbar__row, +main { + max-width: 1200px; + margin: 0 auto; +} + +/*content*/ +body { letter-spacing: 0.2px; } +main { + padding: 90px 0 30px 0; +} + + +/*header*/ +header .mdc-toolbar__title { margin: 0; } +header nav { float: right; margin: 0 !important; } +header nav a { color: #fff !important; min-width: auto !important; } +/*mdc-tab-bar*/ +header nav a { + font-size: .875rem; + line-height: 36px; + font-weight: 500; + letter-spacing: .08929em; + text-transform: uppercase; + text-decoration: none; + padding: 15px 24px; +} +header nav a:hover { + background: #5c7886; +} + + +/*footer*/ +footer { + font-size: 14px; + line-height: 20px; + color: #9e9e9e; + text-align: center; + background: #424242; + padding: 16px 40px; +} +footer a { + color: #9e9e9e; + text-decoration: none; +} +footer .icon-chrome { margin-right: 10px; } +footer .icon-chrome:before { font-size: 17px; } +footer .icon-github:before { + font-size: 20px; position: relative; top: 1px; +} + + +/*custom scrollbars in WebKit*/ +::-webkit-scrollbar, +::-webkit-scrollbar-corner { + height: 10px; + width: 10px; +} +::-webkit-scrollbar-track-piece { + background: #ececec; + border-radius: 2px; +} +::-webkit-scrollbar-thumb { + background: #afbdc3; + border-radius: 6px; +} +::-webkit-scrollbar-thumb:hover { + background-color: #607d8b; +} + + +/*bootstrap callout*/ +.bs-callout { + border: 1px solid #eee; + border-left-width: 5px; + border-left-color: #607d8b; + border-radius: 3px; + background: #fcfcfc; + padding: 20px; + margin: 0 0 20px 0; +} + + +/*button*/ +.s-button { + background-color: #ececec !important; + color: #000 !important; +} +.s-button:before { + background-color: rgba(96, 125, 139, 0.56) !important; +} +.s-button:after { + background-color: rgba(96, 125, 139, 0.56) !important; +} + + +/*misc*/ +h4 { margin: 0 0 10px 0; } + +p { + font-size: 16px; + line-height: 24px; + letter-spacing: 0.2px; +} +p:last-child { + margin-bottom: 0; +} +code { background: #ececec; padding: 5px 8px; border-radius: 3px; } + +.s-label { + font-size: 16px; + letter-spacing: 0.2px; + display: block; + cursor: pointer; + max-width: 400px; + height: 40px; + padding: 5px; + margin: 0 0 8px 0; + border-radius: 3px; +} +.s-label:last-child { + margin-bottom: 0; +} +.s-label.active { + background: #eee; +} +.s-label .mdc-radio { + display: inline-block; +} +.s-label span { + position: relative; + top: -15px; +} + +.s-label em { + font-style: normal; + position: relative; + top: -1px; + padding: 0 10px 0 0; +} diff --git a/css/signin.css b/css/signin.css new file mode 100644 index 0000000..3f97c41 --- /dev/null +++ b/css/signin.css @@ -0,0 +1,90 @@ +@import "https://maxcdn.bootstrapcdn.com/font-awesome/4.5.0/css/font-awesome.min.css"; + +.login__form { + background-color: #263238; + padding: 50px; + border-radius: 10px; +} + +label { + cursor: pointer; + color: #666; +} + +body { + display: flex; + flex-direction: column; + align-items: center; + justify-content: center; + background: #64a4ec; + /* background-color: #263238; */ + font-family: "Lato", sans-serif; + width: 300px; + height: 230px; +} + +h1 { + color: #fff; + text-align: center; + font-size: 40px; + font-weight: 100; + margin: 0 0 30px; + padding: 0; + color: #fff; +} + +.btn { + display: block; + border: 1px solid transparent; + height: 45px; + width: 100%; + border-radius: 5px; + transition: all 0.5s ease; + cursor: pointer; + outline: none; + font-family: "Lato", sans-serif; + font-weight: 600; +} + +.social__login { + display: flex; +} + +.social__login a { + flex: 1; + text-align: center; + text-decoration: none; + line-height: 45px; + margin: 5px; + padding-left: 10px; + padding-right: 10px; + margin-bottom: 20px +} + +.social__login i { + margin-right: 10px; +} + +.btn__google { + background-color: #4285F4; + color: #fff; + font-size: 18px; +} + +.btn__google:hover { + background-color: #fff; + color: #4285F4; + border: 1px solid #4285F4; +} + +.btn__github { + background-color: #333; + color: #fff; + font-size: 18px; +} + +.btn__github:hover { + background-color: #fff; + color: #333; + border: 1px solid #333; +} diff --git a/icons/Jcrop.gif b/icons/Jcrop.gif new file mode 100644 index 0000000..72ea7cc Binary files /dev/null and b/icons/Jcrop.gif differ diff --git a/icons/dark/128x128.png b/icons/dark/128x128.png new file mode 100644 index 0000000..93e48cc Binary files /dev/null and b/icons/dark/128x128.png differ diff --git a/icons/dark/16x16.png b/icons/dark/16x16.png new file mode 100644 index 0000000..04279f2 Binary files /dev/null and b/icons/dark/16x16.png differ diff --git a/icons/dark/19x19.png b/icons/dark/19x19.png new file mode 100644 index 0000000..4301ede Binary files /dev/null and b/icons/dark/19x19.png differ diff --git a/icons/dark/38x38.png b/icons/dark/38x38.png new file mode 100644 index 0000000..8955e83 Binary files /dev/null and b/icons/dark/38x38.png differ diff --git a/icons/dark/48x48.png b/icons/dark/48x48.png new file mode 100644 index 0000000..0c00056 Binary files /dev/null and b/icons/dark/48x48.png differ diff --git a/icons/light/128x128.png b/icons/light/128x128.png new file mode 100644 index 0000000..916772d Binary files /dev/null and b/icons/light/128x128.png differ diff --git a/icons/light/16x16.png b/icons/light/16x16.png new file mode 100644 index 0000000..c3869e4 Binary files /dev/null and b/icons/light/16x16.png differ diff --git a/icons/light/19x19.png b/icons/light/19x19.png new file mode 100644 index 0000000..6c422fb Binary files /dev/null and b/icons/light/19x19.png differ diff --git a/icons/light/38x38.png b/icons/light/38x38.png new file mode 100644 index 0000000..bc7ea82 Binary files /dev/null and b/icons/light/38x38.png differ diff --git a/icons/light/48x48.png b/icons/light/48x48.png new file mode 100644 index 0000000..33e03df Binary files /dev/null and b/icons/light/48x48.png differ diff --git a/icons/pixel.png b/icons/pixel.png new file mode 100644 index 0000000..9da19ea Binary files /dev/null and b/icons/pixel.png differ diff --git a/manifest.json b/manifest.json new file mode 100644 index 0000000..ebdb21f --- /dev/null +++ b/manifest.json @@ -0,0 +1,69 @@ +{ + "manifest_version": 3, + "name" : "Manga Reader", + "version" : "1.2", + "description" : "Manga Reader", + + "icons": { + "16" : "/icons/dark/16x16.png", + "19" : "/icons/dark/19x19.png", + "38" : "/icons/dark/38x38.png", + "48" : "/icons/dark/48x48.png", + "128": "/icons/dark/128x128.png" + }, + + "action": { + "default_icon": { + "16" : "/icons/dark/16x16.png", + "19" : "/icons/dark/19x19.png", + "38" : "/icons/dark/38x38.png", + "48" : "/icons/dark/48x48.png", + "128" : "/icons/dark/128x128.png" + }, + "default_title": "Manga Reader" + }, + + "background" : { + "service_worker": "background.js" + }, + + "options_page": "options.html", + + "web_accessible_resources": [ + { + "matches": [ + "" + ], + "resources": [ + "/icons/Jcrop.gif", + "/icons/pixel.png" + ] + } + ], + + "commands": { + "take-screenshot": { + "description": "Take Screenshot", + "suggested_key": { + "default": "Alt+S" + } + } + }, + + "permissions": [ + "storage", + "scripting", + "activeTab", + "identity", + "contextMenus" + ], + + "oauth2": { + "client_id": "__CLIENT_ID__", + "scopes": [ + "https://www.googleapis.com/auth/userinfo.email", + "https://www.googleapis.com/auth/userinfo.profile" + ] + }, + "key": "__EXTENSION_KEY__" +} diff --git a/options/firebase_config.js b/options/firebase_config.js new file mode 100644 index 0000000..298512c --- /dev/null +++ b/options/firebase_config.js @@ -0,0 +1,29 @@ +// Import the functions you need from the SDKs you need +import { initializeApp } from "firebase/app"; +import { getAnalytics } from "firebase/analytics"; +import 'dotenv/config' + +// TODO: Add SDKs for Firebase products that you want to use +// https://firebase.google.com/docs/web/setup#available-libraries + +// Your web app's Firebase configuration +// For Firebase JS SDK v7.20.0 and later, measurementId is optional +const firebaseConfig = { + apiKey: process.env.FIREBASE_API_KEY, + authDomain: process.env.FIREBASE_AUTH_DOMAIN, + projectId: process.env.FIREBASE_PROJECT_ID, + storageBucket: process.env.FIREBASE_STORAGE_BUCKET, + messagingSenderId: process.env.FIREBASE_MESSAGING_SENDER_ID, + appId: process.env.FIREBASE_APP_ID, + measurementId: process.env.FIREBASE_MEASUREMENT_ID +}; + +// Initialize Firebase +const firebaseApp = initializeApp(firebaseConfig); + +// Trigger index.esm2017.js:87 Refused to load the script 'https://www.googletagmanager.com/gtag/js?l=dataLayer&id=G-28RB8VJQFZ' because it violates the following Content Security Policy directive: "script-src 'self' 'wasm-unsafe-eval' 'inline-speculation-rules' http://localhost:* http://127.0.0.1:*". Note that 'script-src-elem' was not explicitly set, so 'script-src' is used as a fallback. +// const analytics = getAnalytics(firebaseApp); + +export{ + firebaseApp +} \ No newline at end of file diff --git a/options/options.html b/options/options.html new file mode 100644 index 0000000..e2cb75a --- /dev/null +++ b/options/options.html @@ -0,0 +1,31 @@ + + + + Manga Reader + + + + + + +
+
+
+
+

Manga Reader

+
+
+ +
+
+
+
+
+ +
+ +
+ + diff --git a/options/options.js b/options/options.js new file mode 100644 index 0000000..81f9a63 --- /dev/null +++ b/options/options.js @@ -0,0 +1,226 @@ +import { firebaseApp } from './firebase_config' +import { + getAuth, + onAuthStateChanged +} from 'firebase/auth'; +import m from 'mithril'; +import 'material-components-web/dist/material-components-web.min.css'; + +// Auth instance for the current firebaseApp +const auth = getAuth(firebaseApp); + +onAuthStateChanged(auth, user => { + if (user != null) { + console.log('User is logged in!'); + chrome.storage.sync.set({idToken: user.accessToken}) + fetchUserLimit(); + } else { + console.log('User is logged out!'); + chrome.storage.sync.set({idToken: ""}) + window.location.replace('./signin.html'); + } +}); + +document.querySelector('#sign_out').addEventListener('click', () => { + auth.signOut(); +}); + +// Done authentication stuff + +// State used by the options page to determine which options are selected +// and display the selected state in the UI. +var state = { + shortcut: {}, + api: [ + {id: 'gpt', title: 'GPT (For in-depth translation)'}, + {id: 'deepl', title: 'DeepL (For quick translation)'} + ], + source_lang: [ + {id: 'Japanese', title: 'Japanese'}, + {id: 'Korean', title: 'Korean'}, + {id: 'Chinese', title: 'Chinese'} + ], + target_lang: [ + {id: 'English', title: 'English'}, + ], + icon: [ + {id: false, title: 'Dark Icon'}, + {id: true, title: 'Light Icon'} + ], + userLimit: { + requestCount: 0, + limit: 0 + } +} + +// Set the state of the options page based on the current config +chrome.storage.sync.get((config) => { + state.api.forEach((item) => item.checked = item.id === config.api) + state.source_lang.forEach((item) => item.checked = item.id === config.source_lang) + state.target_lang.forEach((item) => item.checked = item.id === config.target_lang) + state.icon.forEach((item) => item.checked = item.id === config.icon) + fetchUserLimit(); + + m.redraw() +}) + +// Get the current keyboard shortcut from the manifest and display it +chrome.commands.getAll((commands) => { + var command = commands.find((command) => command.name === 'take-screenshot') + state.shortcut = command.shortcut + m.redraw() +}) + +// Event handlers for the options page +// These are called when the user interacts with the UI +// If keyboard shortcut is reset, call button function +// If other options are changed (API, icon), call option function +var events = { + option: (name, item) => () => { + state[name].forEach((item) => item.checked = false) + item.checked = true + + chrome.storage.sync.set({[name]: item.id}) + if (name === 'icon') { + const color = item.id ? 'light' : 'dark' + chrome.action.setIcon({ + path: [16, 19, 38, 48, 128].reduce((all, size) => ( + all[size] = `/icons/${color}/${size}x${size}.png`, + all + ), {}) + }) + } + }, + button: (action) => () => { + chrome.tabs.create({url: { + shortcut: 'chrome://extensions/shortcuts', + location: 'chrome://settings/downloads', + }[action]}) + } +} + +var onupdate = (item) => (vnode) => { + if (vnode.dom.classList.contains('active') !== item.checked) { + vnode.dom.classList.toggle('active') + } +} + +m.mount(document.querySelector('main'), { + view: () => [ + m('.bs-callout', + m('h4.mdc-typography--headline5', 'Requests made this month / Your monthly request limit'), + m('p', `${state.userLimit.requestCount} / ${state.userLimit.limit}`) + ), + m('.bs-callout', + m('h4.mdc-typography--headline5', 'Keyboard Shortcut'), + state.shortcut && + m('p', 'Current keyboard shortcut. Refresh after updating to see the updated shorcut.', m('code', state.shortcut)), + !state.shortcut && + m('p', 'No keyboard shortcut set'), + m('button.mdc-button mdc-button--raised s-button', { + onclick: events.button('shortcut') + }, + 'Update' + ) + ), + + m('.bs-callout', + m('h4.mdc-typography--headline5', 'API'), + state.api.map((item) => + m('label.s-label', {onupdate: onupdate(item)}, + m('.mdc-radio', + m('input.mdc-radio__native-control', { + type: 'radio', name: 'api', + checked: item.checked && 'checked', + onchange: events.option('api', item) + }), + m('.mdc-radio__background', + m('.mdc-radio__outer-circle'), + m('.mdc-radio__inner-circle'), + ), + ), + m('span', item.title) + ) + ) + ), + + m('.bs-callout', + m('h4.mdc-typography--headline5', 'Translate From'), + state.source_lang.map((item) => + m('label.s-label', {onupdate: onupdate(item)}, + m('.mdc-radio', + m('input.mdc-radio__native-control', { + type: 'radio', name: 'source_lang', + checked: item.checked && 'checked', + onchange: events.option('source_lang', item) + }), + m('.mdc-radio__background', + m('.mdc-radio__outer-circle'), + m('.mdc-radio__inner-circle'), + ), + ), + m('span', item.title) + ) + ) + ), + + m('.bs-callout', + m('h4.mdc-typography--headline5', 'Translate To'), + state.target_lang.map((item) => + m('label.s-label', {onupdate: onupdate(item)}, + m('.mdc-radio', + m('input.mdc-radio__native-control', { + type: 'radio', name: 'target_lang', + checked: item.checked && 'checked', + onchange: events.option('target_lang', item) + }), + m('.mdc-radio__background', + m('.mdc-radio__outer-circle'), + m('.mdc-radio__inner-circle'), + ), + ), + m('span', item.title) + ) + ) + ), + + m('.bs-callout', + m('h4.mdc-typography--headline5', 'Extension Icon'), + state.icon.map((item) => + m('label.s-label', {onupdate: onupdate(item)}, + m('.mdc-radio', + m('input.mdc-radio__native-control', { + type: 'radio', name: 'icon', + checked: item.checked && 'checked', + onchange: events.option('icon', item) + }), + m('.mdc-radio__background', + m('.mdc-radio__outer-circle'), + m('.mdc-radio__inner-circle'), + ), + ), + m('span', item.title) + ) + ) + ), + ] +}) + +function fetchUserLimit() { + chrome.storage.sync.get(['idToken'], function(result) { + const idToken = result.idToken; + fetch('http://localhost:3000/get-user-limit', { + method: 'GET', + headers: { + 'Authorization': `Bearer ${idToken}` + } + }) + .then(response => response.json()) + .then(data => { + state.userLimit.requestCount = data.request_count; + state.userLimit.limit = data.limit; + m.redraw(); + }) + .catch(error => console.error('Error:', error)); + }); +} diff --git a/options/signin.html b/options/signin.html new file mode 100644 index 0000000..2c5ccab --- /dev/null +++ b/options/signin.html @@ -0,0 +1,24 @@ + + + + + + + Login Page + + + + + + +
+

Manga Reader

+
+ +
+ + + + \ No newline at end of file diff --git a/options/signin.js b/options/signin.js new file mode 100644 index 0000000..6656499 --- /dev/null +++ b/options/signin.js @@ -0,0 +1,96 @@ +import { firebaseApp } from './firebase_config' +import { + getAuth, + onAuthStateChanged, + signInWithCredential, + GoogleAuthProvider, + setPersistence, + browserLocalPersistence +} from 'firebase/auth'; + +// Auth instance for the current firebaseApp +const auth = getAuth(firebaseApp); +setPersistence(auth, browserLocalPersistence) + +function init() { + // Detect auth state + onAuthStateChanged(auth, user => { + if (user != null) { + console.log('Below User is logged in:') + console.log(user) + window.location.replace('./options.html'); + } else { + console.log('No user logged in!'); + } + }); +} +init(); + +document.querySelector('.btn__google').addEventListener('click', () => { + initFirebaseApp() +}); + +function initFirebaseApp() { + // Detect auth state + onAuthStateChanged(auth, user => { + if (user != null) { + console.log('logged in!'); + console.log("current") + console.log(user) + console.log(user.token) + } else { + console.log('No user'); + startSignIn() + } + }); +} + +/** + * Starts the sign-in process. + */ +function startSignIn() { + console.log("started SignIn") + //https://firebase.google.com/docs/auth/web/manage-users + const user = auth.currentUser; + if (user) { + console.log("current") + console.log(user) + auth.signOut(); + } else { + console.log("proceed") + startAuth(true); + } +} + +/** + * Start the auth flow and authorizes to Firebase. + * @param{boolean} interactive True if the OAuth flow should request with an interactive mode. + */ +function startAuth(interactive) { + console.log("Auth trying") + chrome.identity.getAuthToken({ interactive: true }, function (token) { + //Token: This requests an OAuth token from the Chrome Identity API. + if (chrome.runtime.lastError && !interactive) { + console.log('It was not possible to get a token programmatically.'); + } else if (chrome.runtime.lastError) { + console.error(chrome.runtime.lastError); + } else if (token) { + // Follows: https://firebase.google.com/docs/auth/web/google-signin + // Authorize Firebase with the OAuth Access Token. + // console.log("TOKEN:") + // console.log(token) + // Builds Firebase credential with the Google ID token. + const credential = GoogleAuthProvider.credential(null, token); + signInWithCredential(auth, credential).then((result) => { + console.log("Success!!!") + console.log(result) + }).catch((error) => { + // You can handle errors here + console.log(error) + }); + } else { + console.error('The OAuth token was null'); + } + }); +} + diff --git a/package.json b/package.json new file mode 100644 index 0000000..16b5d87 --- /dev/null +++ b/package.json @@ -0,0 +1,34 @@ +{ + "scripts": { + "build": "webpack --watch --config webpack.development.js", + "release": "webpack --config webpack.production.js" + }, + "devDependencies": { + "clean-webpack-plugin": "^4.0.0", + "copy-webpack-plugin": "^10.2.3", + "html-webpack-plugin": "^5.5.0", + "webpack": "^5.88.2", + "webpack-cli": "^4.10.0" + }, + "dependencies": { + "@material/button": "^14.0.0", + "@material/elevation": "^14.0.0", + "@material/radio": "^14.0.0", + "@material/theme": "^14.0.0", + "@material/toolbar": "^2.3.0", + "@material/typography": "^14.0.0", + "babel-loader": "^9.1.3", + "css-loader": "^6.8.1", + "dotenv": "^16.3.1", + "dotenv-webpack": "^8.0.1", + "file-loader": "^6.2.0", + "firebase": "^9.6.5", + "firebaseui": "^6.0.0", + "jquery": "^3.7.0", + "jquery-jcrop": "^0.9.15", + "material-components-web": "^14.0.0", + "mithril": "^2.2.2", + "node-polyfill-webpack-plugin": "^2.0.1", + "style-loader": "^3.3.3" + } +} diff --git a/server/.env_template b/server/.env_template new file mode 100644 index 0000000..9479949 --- /dev/null +++ b/server/.env_template @@ -0,0 +1,3 @@ +OPENAI_API_KEY= +DEEPL_API_KEY= +PINECONE_KEY= \ No newline at end of file diff --git a/server/.gcloudignore b/server/.gcloudignore new file mode 100644 index 0000000..52535c3 --- /dev/null +++ b/server/.gcloudignore @@ -0,0 +1,25 @@ +# This file specifies files that are *not* uploaded to Google Cloud +# using gcloud. It follows the same syntax as .gitignore, with the addition of +# "#!include" directives (which insert the entries of the given .gitignore-style +# file at that point). +# +# For more information, run: +# $ gcloud topic gcloudignore +# +.gcloudignore +# If you would like to upload your .git directory, .gitignore file or files +# from your .gitignore file, remove the corresponding line +# below: +.git +.gitignore +.venv +node_modules + +# Python pycache: +__pycache__/ +# Ignored by the build system +/setup.cfg + +# not yet +populate_vectorstore.py +query_test.py \ No newline at end of file diff --git a/server/__init__.py b/server/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/server/app.yaml b/server/app.yaml new file mode 100644 index 0000000..9d782c9 --- /dev/null +++ b/server/app.yaml @@ -0,0 +1,9 @@ +runtime: python39 +instance_class: F4_1G +entrypoint: gunicorn -b :$PORT main:app +automatic_scaling: + min_instances: 1 + +handlers: +- url: /.* + script: auto diff --git a/server/download_model.py b/server/download_model.py new file mode 100644 index 0000000..df9c410 --- /dev/null +++ b/server/download_model.py @@ -0,0 +1,14 @@ +from transformers import AutoFeatureExtractor, AutoTokenizer, VisionEncoderDecoderModel + +pretrained_model_name_or_path='kha-white/manga-ocr-base' + +# Download components +feature_extractor = AutoFeatureExtractor.from_pretrained(pretrained_model_name_or_path) +tokenizer = AutoTokenizer.from_pretrained(pretrained_model_name_or_path) +model = VisionEncoderDecoderModel.from_pretrained(pretrained_model_name_or_path) + +# Save components to local directory +save_directory = "./model" +feature_extractor.save_pretrained(save_directory) +tokenizer.save_pretrained(save_directory) +model.save_pretrained(save_directory) \ No newline at end of file diff --git a/server/imageDataUrl b/server/imageDataUrl new file mode 100644 index 0000000..db89acd --- /dev/null +++ b/server/imageDataUrl @@ -0,0 +1 @@ +// const imageDataUrl = ''; \ No newline at end of file diff --git a/server/main.py b/server/main.py new file mode 100644 index 0000000..478f571 --- /dev/null +++ b/server/main.py @@ -0,0 +1,124 @@ +from flask import Flask, request, jsonify +from flask_cors import CORS +from functools import wraps +from flask import request, jsonify +import firebase_admin +from firebase_admin import auth, credentials +from requests import exceptions +from dotenv import load_dotenv +from services import UsersService +from services import OCRService +from services import TranslationService +import os + +load_dotenv() +hard_limit = 1000 + +# Initialize Firebase Admin SDK with your service account credentials +cred = credentials.Certificate('firebaseServiceAccountKey.json') +firebase_admin.initialize_app(cred) + +# Decorator function to authenticate API requests +def authenticate(func): + @wraps(func) + def wrapper(*args, **kwargs): + # Get the authorization header from the request + auth_header = request.headers.get('Authorization') + + if not auth_header: + return jsonify({'error': 'Unauthorized'}), 401 + + try: + # Extract the ID token from the authorization header + id_token = auth_header.split(' ')[1] + + # Verify the ID token using Firebase Authentication + decoded_token = auth.verify_id_token(id_token, check_revoked=True) + + # Add the user's UID to the request context + kwargs['user_id'] = decoded_token['uid'] + + return func(*args, **kwargs) + except (auth.InvalidIdTokenError, IndexError, ValueError, exceptions.RequestException) as e: + print(e) + return jsonify({'error': 'Token might have expired. Please sign in again.'}), 401 + + return wrapper + +# Replace with your actual keys + +app = Flask(__name__) +CORS(app) +users_service = UsersService(hard_limit=hard_limit) +ocr_service = OCRService() +translation_serivce = TranslationService(os.getenv("OPENAI_API_KEY"), os.getenv("DEEPL_API_KEY")) + +@app.route("/translate-text", methods=["POST"]) +@authenticate +def translate_text(user_id): + source_lang = request.args.get('source_lang') + target_lang = request.args.get('target_lang', 'English') # Added target_lang argument + request_count, limit = users_service.get_request_count(user_id) + if request_count > limit: + return jsonify({"error": f"You have exceeded your monthly request limit: {str(limit)}"}), 403 + users_service.increment_request_count(user_id) + text = request.json.get('text') + if source_lang not in ["Japanese", "Korean", "Chinese"]: + return jsonify({"error": "Unsupported language"}), 400 + api = request.args.get('api') + if api == "gpt": + try: + translation = translation_serivce.call_gpt(text, source_lang, target_lang) # Pass target_lang to the service + except ValueError as e: + return jsonify({"error": str(e)}), 400 + elif api == "deepl": + try: + translation = translation_serivce.call_deepl(text, source_lang, target_lang) # Pass target_lang to the service + except ValueError as e: + return jsonify({"error": str(e)}), 400 + else: + return jsonify({"error": "Invalid API"}), 400 + users_service.store_request_data(user_id, text, translation, "text", api) + return jsonify({"translation": translation}) + +@app.route("/translate-img", methods=["POST"]) +@authenticate +def translate_img(user_id): + source_lang = request.args.get('source_lang') + target_lang = request.args.get('target_lang', 'English') # Added target_lang argument + request_count, limit = users_service.get_request_count(user_id) + if request_count > limit: + return jsonify({"error": f"You have exceeded your monthly request limit: {str(limit)}"}), 403 + users_service.increment_request_count(user_id) + image_data_url = request.json.get('imageDataUrl') + print(source_lang) + if source_lang not in ["Japanese", "Korean", "Chinese"]: + return jsonify({"error": "Unsupported language"}), 400 + try: + text = ocr_service.annotate_image(image_data_url, source_lang) + except ValueError as e: + return jsonify({"error": str(e)}), 400 + api = request.args.get('api') + if api == "gpt": + try: + translation = translation_serivce.call_gpt(text, source_lang, target_lang) + except ValueError as e: + return jsonify({"error": str(e)}), 400 + elif api == "deepl": + try: + translation = translation_serivce.call_deepl(text, source_lang, target_lang) + except ValueError as e: + return jsonify({"error": str(e)}), 400 + else: + return jsonify({"error": "Invalid API"}), 400 + users_service.store_request_data(user_id, text, translation, "image", api) + return jsonify({"translation": translation, "original": text}) + +@app.route("/get-user-limit", methods=["GET"]) +@authenticate +def get_user_limit(user_id): + request_count, limit = users_service.get_request_count(user_id) + return jsonify({"request_count": request_count, "limit": limit}) + +if __name__ == "__main__": + app.run(port=3000) \ No newline at end of file diff --git a/server/populate_vectorstore.py b/server/populate_vectorstore.py new file mode 100644 index 0000000..17fdccd --- /dev/null +++ b/server/populate_vectorstore.py @@ -0,0 +1,62 @@ +from llama_index import VectorStoreIndex, download_loader, StorageContext +from llama_index.vector_stores import PineconeVectorStore + +"""Simple reader that reads wikipedia.""" +from typing import Any, List + +from llama_index.readers.base import BaseReader +from llama_index.schema import Document + +from dotenv import load_dotenv +import os +import openai +import pinecone + +load_dotenv() +openai.api_key = os.environ["OPENAI_API_KEY"] + +class JaWikipediaReader(BaseReader): + """Wikipedia reader. + + Reads a page. + + """ + + def __init__(self) -> None: + """Initialize with parameters.""" + try: + import wikipedia # noqa: F401 + except ImportError: + raise ImportError( + "`wikipedia` package not found, please run `pip install wikipedia`" + ) + + def load_data(self, pages: List[str], **load_kwargs: Any) -> List[Document]: + """Load data from the input directory. + + Args: + pages (List[str]): List of pages to read. + + """ + import wikipedia + wikipedia.set_lang("ja") + results = [] + for page in pages: + page_content = wikipedia.page(page, **load_kwargs).content + results.append(Document(text=page_content)) + return results + +WikipediaReader = download_loader("WikipediaReader") +loader = JaWikipediaReader() +documents = loader.load_data(pages=['ONE_PIECE', 'ONE_PIECEの登場人物一覧', 'ONE_PIECEの用語一覧', 'ONE_PIECEの地理']) + +# init pinecone +pinecone.init(api_key=os.environ["OPENAI_API_KEY"], environment="asia-southeast1-gcp-free") +# pinecone.create_index("manga-reader", dimension=1536, metric="cosine", pod_type="p1") + +# construct vector store and customize storage context +storage_context = StorageContext.from_defaults( + vector_store = PineconeVectorStore(pinecone.Index("manga-reader")) +) +index = VectorStoreIndex.from_documents(documents, storage_context=storage_context) + diff --git a/server/query_test.py b/server/query_test.py new file mode 100644 index 0000000..06e70b3 --- /dev/null +++ b/server/query_test.py @@ -0,0 +1,140 @@ + +import pinecone +from llama_index import VectorStoreIndex +from llama_index.vector_stores import PineconeVectorStore +from dotenv import load_dotenv +import os +import openai +import pinecone +from llama_index.query_engine import RetrieverQueryEngine +from main import call_gpt +load_dotenv() +openai.api_key = os.environ["OPENAI_API_KEY"] +pinecone.init(api_key=os.environ["PINECONE_KEY"], environment="asia-southeast1-gcp-free") + +# vector_store = PineconeVectorStore(pinecone.Index("manga-reader")) +# index = VectorStoreIndex.from_vector_store(vector_store=vector_store) +# retriever = index.as_retriever(similarity_top_k=5, search_type="mmr") +# documents = retriever.retrieve("茶ひげ海賊団") +# # print([doc.node.text for doc in documents]) +# stuff_docs = [doc.node.text + "\n" for doc in documents] +# print(stuff_docs) +# query_engine = RetrieverQueryEngine.from_args(retriever=index.as_retriever(similarity_top_k=0), response_mode="compact") + +# messages=[ +# {"role": "user", "content": f""" +# You are a robotic translator who has mastered all languages. You provide the translation and breakdown +# of the phrase or a word directly without trying to engage in a conversation. When given a phrase or word to be +# translated, you first provide the orignal phrase or word to be translated, followed by the direc translation in English +# and only the direct translation, +# followed by the breakdown of the phrase into compound words or loan words if necessary and explain their definitions. + +# Use the attached context to first check if the phrase contains a name or a place from a fictional universe. If so, point +# out that the phrase to be translated has a name or a place. +# context: +# {stuff_docs} + +# To be translated: 茶ひげーお前ら捕まっちつかまうのかまえ +# Translation: +# """} +# ] + +# print(call_gpt(messages)["content"]) + + + +# ####### check if a sentence contains a name or place ####### + + +# messages=[ +# {"role": "user", "content": f""" +# You are a robotic translator who has mastered all languages. You provide the translation and breakdown +# of the phrase or a word directly without trying to engage in a conversation. Sometimes, sentences or phrases to be +# translated might contains a name or a place from a fictional universe. You first check if the phrase might +# contain a name or a place from a fictional universe. Say yes or no. + +# For example, "茶ひげ" in "茶ひげーお前ら捕まっちつかまうのかまえ" could be a name. + +# To be translated: "好きな食べ物はアーモンド" +# Yes or no:"""} +# ] +# print(call_gpt(messages)["content"]) + + + +####### check if a phrase is broken up or contains extra characters ####### +# Always returns valid currently + +# messages=[ +# {"role": "user", "content": f""" +# You are a robotic translator who has mastered all languages. You provide the translation and breakdown +# of the phrase or a word directly without trying to engage in a conversation. Sometimes, sentences or phrases +# that users provide might have omitted trailing or preceding characters. You first check if the phrase might +# be problematic. Say "incomplete", "extra", or "valid". + +# For example, in the case where "茶ひげーお前ら捕まっちつかまうのかまえ" is to be translated +# user might have only provided "茶ひげーお前ら捕まっちつかまうのか" due to a mistake. Say "incomplete" in this case. + +# in the case where "研究所までの運搬役" is to be translated +# user might have only provided 研究所までの運搬役にさ" due to a mistake. Say "extra" in this case. + +# To be translated: 僻地に存 +# validity check:"""} +# ] +# print(call_gpt(messages)["content"]) + +###### check if a phrase is broken up or contains extra characters ####### + +messages=[ + {"role": "user", "content": f""" +You are a robotic translator who is great at translating Japanese to any other languages. Sometimes, users +copy a Japnaese phrase from an image using OCR, but the Japanese text might have furigana annotations. +Your task is to identify furigana and remove it from the text provided by the human. + +For example, in the case where "スモーカー先生!!!" is to be translated +user might have only provided "スモーカー先生!!!せんせいえ". Say "スモーカー先生!!!" in this case. + +To be translated: 僻地に存 +validity check:"""} + ] +print(call_gpt(messages)["content"]) + + +############# langchain stuff ################ + +# from langchain.vectorstores import Pinecone +# from langchain.embeddings.openai import OpenAIEmbeddings +# from langchain.chat_models import ChatOpenAI +# from langchain.chains import RetrievalQA +# from langchain.prompts import PromptTemplate +# embeddings = OpenAIEmbeddings() + +# docsearch = Pinecone.from_existing_index("manga-reader", embedding=embeddings) +# retriever = docsearch.as_retriever(search_type="similarity", k=5) # or mmr +# can also use docsearch.similarity_search_with_score() to directly get the documents +# retriever.search_kwargs["k"] = 5 +# print(docsearch.similarity_search_with_score("茶ひげーお前ら捕まっちつかまうのかまえ")) + +# template = """ +# You are a robotic translator who has mastered all languages. You provide the translation and breakdown +# of the phrase or a word directly without trying to engage in a conversation. When given a phrase or word to be +# translated, you first provide the orignal phrase or word to be translated, followed by the direc translation in English +# and only the direct translation, +# followed by the breakdown of the phrase into compound words or loan words if necessary and explain their definitions. + +# You may use the attached context, in case the phrase contains a name or a place from a fictional universe. +# context: +# {context} + +# To be translated: {question} +# Translation:""" +# PROMPT = PromptTemplate(input_variables=["context", "question"], template=template) +# model = ChatOpenAI(model="gpt-3.5-turbo") +# qa = RetrievalQA.from_chain_type( +# llm=model, chain_type="stuff", +# retriever=retriever, +# chain_type_kwargs={"prompt": PROMPT}, +# verbose=True) + +# output = qa({"query": "茶ひげーお前ら捕まっちつかまうのかまえ"}) +# print(output) \ No newline at end of file diff --git a/server/requirements.txt b/server/requirements.txt new file mode 100644 index 0000000..d5c724f --- /dev/null +++ b/server/requirements.txt @@ -0,0 +1,94 @@ +aiohttp==3.8.4 +aiosignal==1.3.1 +async-timeout==4.0.2 +attrs==23.1.0 +blinker==1.6.2 +cachecontrol==0.13.1 +cachetools==5.3.1 +certifi==2023.5.7 +cffi==1.15.1 +charset-normalizer==3.2.0 +click==8.1.4 +cmake==3.26.4 +cryptography==41.0.1 +filelock==3.12.2 +fire==0.5.0 +firebase-admin==6.2.0 +Flask==2.3.2 +Flask-Cors==4.0.0 +frozenlist==1.3.3 +fsspec==2023.6.0 +fugashi==1.2.1 +google-api-core==2.11.1 +google-api-python-client==2.92.0 +google-auth==2.21.0 +google-auth-httplib2==0.1.0 +google-cloud-core==2.3.3 +google-cloud-firestore==2.11.1 +google-cloud-storage==2.10.0 +google-cloud-vision==3.4.4 +google-crc32c==1.5.0 +google-resumable-media==2.5.0 +googleapis-common-protos==1.59.1 +grpcio==1.56.0 +grpcio-status==1.56.0 +gunicorn==20.1.0 +httplib2==0.22.0 +huggingface-hub==0.16.4 +idna==3.4 +importlib-metadata==6.8.0 +itsdangerous==2.1.2 +jaconv==0.3.4 +Jinja2==3.1.2 +lit==16.0.6 +loguru==0.7.0 +manga-ocr==0.1.10 +MarkupSafe==2.1.3 +mpmath==1.3.0 +msgpack==1.0.5 +multidict==6.0.4 +networkx==3.1 +numpy==1.25.1 +nvidia-cublas-cu11==11.10.3.66 +nvidia-cuda-cupti-cu11==11.7.101 +nvidia-cuda-nvrtc-cu11==11.7.99 +nvidia-cuda-runtime-cu11==11.7.99 +nvidia-cudnn-cu11==8.5.0.96 +nvidia-cufft-cu11==10.9.0.58 +nvidia-curand-cu11==10.2.10.91 +nvidia-cusolver-cu11==11.4.0.1 +nvidia-cusparse-cu11==11.7.4.91 +nvidia-nccl-cu11==2.14.3 +nvidia-nvtx-cu11==11.7.91 +openai==0.27.8 +packaging==23.1 +Pillow==10.0.0 +proto-plus==1.22.3 +protobuf==4.23.4 +pyasn1==0.5.0 +pyasn1-modules==0.3.0 +pycparser==2.21 +PyJWT==2.7.0 +pyparsing==3.1.0 +pyperclip==1.8.2 +python-dotenv==1.0.0 +PyYAML==6.0 +regex==2023.6.3 +requests==2.31.0 +rsa==4.9 +safetensors==0.3.1 +six==1.16.0 +sympy==1.12 +termcolor==2.3.0 +tokenizers==0.13.3 +torch==2.0.1 +tqdm==4.65.0 +transformers==4.30.2 +triton==2.0.0 +typing-extensions==4.7.1 +unidic-lite==1.0.8 +uritemplate==4.1.1 +urllib3==1.26.16 +Werkzeug==2.3.6 +yarl==1.9.2 +zipp==3.16.0 diff --git a/server/services/__init__.py b/server/services/__init__.py new file mode 100644 index 0000000..cb7aa29 --- /dev/null +++ b/server/services/__init__.py @@ -0,0 +1,3 @@ +from .ocr_service import OCRService +from .translation_service import TranslationService +from .users_service import UsersService \ No newline at end of file diff --git a/server/services/ocr_service.py b/server/services/ocr_service.py new file mode 100644 index 0000000..e5c2a63 --- /dev/null +++ b/server/services/ocr_service.py @@ -0,0 +1,50 @@ +import io +import re +import base64 +from PIL import Image +from manga_ocr import MangaOcr +from google.cloud import vision + +class OCRService: + def __init__(self): + self.mocr = MangaOcr(pretrained_model_name_or_path='./model') + print("using mocr!!!!!!!!!!!") + + self.client = vision.ImageAnnotatorClient() + + def annotate_image(self, image_data_url, source_lang): + if source_lang == "Japanese": + return self.annotate_image_manga_ocr(image_data_url) + elif source_lang == "Korean" or source_lang == "Chinese": + return self.annotate_image_google_vision(image_data_url, source_lang) + else: + raise ValueError("Not Supported") + + def annotate_image_google_vision(self, image_data_url, source_lang): + image_content = base64.b64decode(image_data_url.split(',')[1]) + image = vision.Image(content=image_content) + + response = self.client.text_detection(image=image) + texts = response.text_annotations + text = texts[0].description if len(texts) > 0 else '' + + print(text) + if source_lang == "Japanese" or source_lang == "Chinese": + text = text.replace("\n", "") + else: + text = text.replace("\n", " ") + + return text + + def annotate_image_manga_ocr(self, image_data_url): + # Extract the base64-encoded image data from the URL + image_data = re.sub('^data:image/.+;base64,', '', image_data_url) + + # Decode the base64-encoded image data + decoded_image_data = io.BytesIO(base64.b64decode(image_data)) + + # Open the image using PIL + image = Image.open(decoded_image_data) + + text = self.mocr(image) + return text \ No newline at end of file diff --git a/server/services/translation_service.py b/server/services/translation_service.py new file mode 100644 index 0000000..daded3d --- /dev/null +++ b/server/services/translation_service.py @@ -0,0 +1,55 @@ +import openai +import requests # make sure you have this line +from flask import request, jsonify + +class TranslationService: + def __init__(self, openai_api_key, deepl_api_key): + openai.api_key = openai_api_key + self.deepl_api_key = deepl_api_key + self.lang_map = {"Japanese": "JA", "Korean": "KO", "Chinese": "ZH", "English": "EN"} # Added 'EN' mapping for English and 'ZH' mapping for Chinese + + def call_gpt(self, text, source_lang, target_lang): + prompt = f'translate the {source_lang} phrase or word "{text}" to {target_lang}.' # Updated to handle long format language names + messages=[ + {"role": "system", "content": """ + You are a robotic translator who has mastered all languages. You provide the translation and breakdown + of the phrase or a word directly without trying to engage in a conversation. When given a phrase or word to be + translated, you first provide the orignal phrase or word to be translated, followed by the direc translation in English + and only the direct translation, + followed by the breakdown of the phrase into compound words or loan words if necessary and explain their definitions."""}, + {"role": "user", "content": prompt} + ] + completion = openai.ChatCompletion.create( + model="gpt-3.5-turbo-0613", + messages=messages, + temperature=0.2, + ) + + print(completion.choices[0]['message']) + return completion.choices[0]['message']['content'] + + def call_deepl(self, text, source_lang, target_lang): + target_lang = self.lang_map[target_lang] # Updated to use language map + + url = "https://api-free.deepl.com/v2/translate" + + data = { + "text": text, + "source_lang": self.lang_map.get(source_lang), + "target_lang": target_lang + } + + headers = { + "Authorization": f"DeepL-Auth-Key {self.deepl_api_key}", + "Content-Type": "application/x-www-form-urlencoded;charset=UTF-8" + } + + response = requests.post(url, data=data, headers=headers) + print(response) + json_resp = response.json() + + if "message" in json_resp: + return jsonify({"translation": f"DeepL message: {json_resp['message']}"}) + + print(json_resp['translations'][0]['text']) + return json_resp['translations'][0]['text'] \ No newline at end of file diff --git a/server/services/users_service.py b/server/services/users_service.py new file mode 100644 index 0000000..b90b6f3 --- /dev/null +++ b/server/services/users_service.py @@ -0,0 +1,65 @@ +import datetime +from google.cloud import firestore + +class UsersService: + + def __init__(self, hard_limit): + self.db = firestore.Client() + self.hard_limit = hard_limit + + # Function to increment request count for a user + def increment_request_count(self, user_id): + # Get current month and year + current_month = datetime.datetime.now().strftime("%Y-%m") + + # Define the document path for the user's request count + doc_path = f"users/{user_id}/monthlyCounts/{current_month}" + + # Increment the request count document + doc_ref = self.db.document(doc_path) + doc_ref.set({"count": firestore.Increment(1)}, merge=True) + + # Function to store request data in Firestore + def store_request_data(self, user_id, text, translation, type, api): + # Define the Firestore collection and document paths + collection_path = f"users/{user_id}/requests" + document_path = self.db.collection(collection_path).document() + + # Create a new document with autogenerated ID + request_doc_ref = self.db.document(document_path.path) + request_data = { + "text": text, + "type": type, + "api": api, + "response": translation + } + # Store the request data in the document + request_doc_ref.set(request_data) + + # Function to retrieve request count for a user + def get_request_count(self, user_id): + current_month = datetime.datetime.now().strftime("%Y-%m") + doc_path = f"users/{user_id}/monthlyCounts/{current_month}" + doc_ref = self.db.document(doc_path) + doc_snapshot = doc_ref.get() + + custom_limit = self.get_custom_limit(user_id) + limit = custom_limit if custom_limit is not None else self.hard_limit + + if doc_snapshot.exists: + request_count = doc_snapshot.get("count") + return request_count, limit + else: + return 0, limit + + def get_custom_limit(self, user_id): + current_month = datetime.datetime.now().strftime("%Y-%m") + doc_path = f"users/{user_id}/customLimit/{current_month}" + doc_ref = self.db.document(doc_path) + doc_snapshot = doc_ref.get() + + if doc_snapshot.exists: + custom_limit = doc_snapshot.get("limit") + return custom_limit + else: + return None \ No newline at end of file diff --git a/server/test_image.jpg b/server/test_image.jpg new file mode 100644 index 0000000..9da19ea Binary files /dev/null and b/server/test_image.jpg differ diff --git a/webpack.common.js b/webpack.common.js new file mode 100644 index 0000000..3cb8823 --- /dev/null +++ b/webpack.common.js @@ -0,0 +1,107 @@ +const path = require('path'); +const HtmlWebpackPlugin = require('html-webpack-plugin'); +const { CleanWebpackPlugin } = require('clean-webpack-plugin'); +const CopyWebpackPlugin = require('copy-webpack-plugin'); +var webpack = require('webpack'); +const NodePolyfillPlugin = require('node-polyfill-webpack-plugin') +require('dotenv').config({ path: './.env' }); + +module.exports = { + // Note: + // Chrome MV3 no longer allowed remote hosted code + // Using module bundlers we can add the required code for your extension + // Any modular script should be added as entry point + entry: { + options: './options/options.js', + firebase_config: './options/firebase_config.js', + signin: './options/signin.js', + content: './content/content.js', + }, + plugins: [ + // adds jquery to all modules. May not be necessary but need + // to find a way to add it before adding content.js + new webpack.ProvidePlugin({ + $: "jquery", + jQuery: "jquery", + }), + new CleanWebpackPlugin({ cleanStaleWebpackAssets: false }), + new HtmlWebpackPlugin({ + template: path.join(__dirname, "options", "signin.html"), + filename: "signin.html", + chunks: ["signin"] // This is script from entry point + }), + // Note: you can add as many new HtmlWebpackPlugin objects + // filename: being the html filename + // chunks: being the script src + // if the script src is modular then add it as the entry point above + new HtmlWebpackPlugin({ + template: path.join(__dirname, "options", "options.html"), + filename: "options.html", + chunks: ["options"] // This is script from entry point + }), + + // Note: This is to copy any remaining files to bundler + new CopyWebpackPlugin({ + patterns: [ + { + from: 'manifest.json', + to: 'manifest.json', + transform(content) { + const replacedContent = content + .toString() + .replace('__EXTENSION_KEY__', process.env.EXTENSION_KEY) + .replace('__CLIENT_ID__', process.env.CLIENT_ID); + return Buffer.from(replacedContent); + }, + }, { from: './icons/**' }, + { from: './css/*' }, + { from: '.env'}, + { + from: './background/background.js', + to: 'background.js', + transform(content) { + const replacedContent = content + .toString() + .replace('__BACKEND_URL__', process.env.BACKEND_URL); + return Buffer.from(replacedContent); + }, + }, + ], + }), + + // The following two are required for dotenv + new NodePolyfillPlugin(), + new webpack.DefinePlugin({ + "process.env": JSON.stringify(process.env), + }), + ], + output: { + // chrome load uppacked extension looks for files under dist/* folder + filename: '[name].js', + path: path.resolve(__dirname, 'dist') + }, + module: { + rules: [ + { + test: /\.css$/i, + use: ['style-loader', 'css-loader'], + }, + { + test: /\.(gif|png|jpe?g|svg)$/i, + use: [ + { + loader: 'file-loader', + options: { + outputPath: 'images', + }, + }, + ], + }, + { + test: /\.js$/, + exclude: /node_modules/, + use: 'babel-loader' + }, + ], + }, +}; diff --git a/webpack.development.js b/webpack.development.js new file mode 100644 index 0000000..2ba7544 --- /dev/null +++ b/webpack.development.js @@ -0,0 +1,7 @@ +const { merge } = require('webpack-merge'); +const common = require('./webpack.common.js'); + +module.exports = merge(common, { + mode: 'development', + devtool: 'cheap-module-source-map', +}); \ No newline at end of file diff --git a/webpack.production.js b/webpack.production.js new file mode 100644 index 0000000..812b091 --- /dev/null +++ b/webpack.production.js @@ -0,0 +1,6 @@ +const { merge } = require('webpack-merge'); +const common = require('./webpack.common.js'); + +module.exports = merge(common, { + mode: 'production', +}); \ No newline at end of file