Compare commits

...

11 Commits

Author SHA1 Message Date
ColonelParrot
e9af1941e6
Update README.md 2025-03-13 23:21:51 -04:00
ColonelParrot
ce85c440ad fix scaling issue for logo 2025-03-13 23:12:58 -04:00
ColonelParrot
3f4b902c48 Bump to v1.4.0 2025-02-18 11:38:31 -05:00
ColonelParrot
1fa3133944 optimize extractPaper by saving compute when cornerPoints supplied 2025-02-18 11:36:46 -05:00
ColonelParrot
58b26c34c7 fix skipped tests with before() 2025-02-18 11:36:12 -05:00
ColonelParrot
b4e68b12c9 Update canvas, jsdom and mocha to resolve audit warnings 2025-02-02 00:09:54 -05:00
ColonelParrot
57bf40f3fb bump to v1.3.3 2025-02-01 21:10:04 -05:00
ColonelParrot
c8e6ab035d Add null check in extractPaper 2025-02-01 21:06:30 -05:00
ColonelParrot
e2af210cf2 port bugfix to node 2025-02-01 21:01:53 -05:00
Leon Scherer
f0428d79ca fix: sending invalid index to opencv 2025-02-01 20:23:48 -05:00
ColonelParrot
a447c90039 add wiki link 2025-01-29 13:01:03 -05:00
7 changed files with 1498 additions and 2144 deletions

128
README.md
View File

@ -1,127 +1 @@
<p align="center"> # this repository has been moved to [puffinsoft/jscanify](https://github.com/puffinsoft/jscanify)
<img src="docs/images/logo-github.png" height="150">
</p>
<p align="center">
<a href="https://www.jsdelivr.com/package/gh/ColonelParrot/jscanify"><img src="https://data.jsdelivr.com/v1/package/gh/ColonelParrot/jscanify/badge"></a>
<a href="https://cdnjs.com/libraries/jscanify"><img src="https://img.shields.io/cdnjs/v/jscanify"></a>
<a href="https://npmjs.com/package/jscanify"><img src="https://badgen.net/npm/dw/jscanify"></a>
<br />
<a href="https://github.com/puffinsoft/jscanify/blob/master/LICENSE"><img src="https://img.shields.io/github/license/puffinsoft/jscanify.svg"></a>
<a href="https://npmjs.com/package/jscanify"><img src="https://badgen.net/npm/v/jscanify"></a>
</p>
<p align="center">
<a href="https://nodei.co/npm/jscanify/"><img src="https://nodei.co/npm/jscanify.png"></a>
</p>
<p align="center">
Powered with <a href="https://docs.opencv.org/3.4/d5/d10/tutorial_js_root.html">opencv.js</a><br/>
Supports the web, NodeJS, <a href="https://github.com/ColonelParrot/react-scanify-demo">React</a>, and others.
<br/>
Available on <a href="https://www.npmjs.com/package/jscanify">npm</a> or via <a href="https://www.jsdelivr.com/package/gh/ColonelParrot/jscanify">cdn</a><br/>
</p>
**Features**:
- paper detection & highlighting
- paper scanning with distortion correction
> [!IMPORTANT]
> 🎉 _jscanify v1.3.0_ has just been released! **Same API, better results.** See the [release](https://github.com/puffinsoft/jscanify/releases/tag/v1.3.0) to see the difference! 🎉
- 🆕 glare suppression
- 🆕 multi-colored paper support
<hr />
<img src="docs/images/github-explanation-long.png" />
<hr/>
## Quickstart
> **Developers Note**: you can now use the [jscanify debugging tool](https://colonelparrot.github.io/jscanify/tester.html) to observe the result (highlighting, extraction) on test images.
### Import
npm:
```js
$ npm i jscanify
import jscanify from 'jscanify'
```
cdn:
```html
<script src="https://docs.opencv.org/4.7.0/opencv.js" async></script>
<!-- warning: loading OpenCV can take some time. Load asynchronously -->
<script src="https://cdn.jsdelivr.net/gh/ColonelParrot/jscanify@master/src/jscanify.min.js"></script>
```
> **Note**: jscanify on NodeJS is slightly different. See [wiki: use on NodeJS](https://github.com/ColonelParrot/jscanify/wiki#use-on-nodejs).
### Highlight Paper in Image
```html
<img src="/path/to/your/image.png" id="image" />
```
```js
const scanner = new jscanify();
image.onload = function () {
const highlightedCanvas = scanner.highlightPaper(image);
document.body.appendChild(highlightedCanvas);
};
```
### Extract Paper
```js
const scanner = new jscanify();
const paperWidth = 500;
const paperHeight = 1000;
image.onload = function () {
const resultCanvas = scanner.extractPaper(image, paperWidth, paperHeight);
document.body.appendChild(resultCanvas);
};
```
### Highlighting Paper in User Camera
The following code continuously reads from the user's camera and highlights the paper:
```html
<video id="video"></video> <canvas id="canvas"></canvas>
<!-- original video -->
<canvas id="result"></canvas>
<!-- highlighted video -->
```
```js
const scanner = new jscanify();
const canvasCtx = canvas.getContext("2d");
const resultCtx = result.getContext("2d");
navigator.mediaDevices.getUserMedia({ video: true }).then((stream) => {
video.srcObject = stream;
video.onloadedmetadata = () => {
video.play();
setInterval(() => {
canvasCtx.drawImage(video, 0, 0);
const resultCanvas = scanner.highlightPaper(canvas);
resultCtx.drawImage(resultCanvas, 0, 0);
}, 10);
};
});
```
To export the paper to a PDF, see [here](https://stackoverflow.com/questions/23681325/convert-canvas-to-pdf)
### Notes
- for optimal paper detection, the paper should be placed on a flat surface with a solid background color
- we recommend wrapping your code using `jscanify` in a window `load` event listener to ensure OpenCV is loaded

View File

@ -32,7 +32,7 @@
fill="black" class="octo-body"></path> fill="black" class="octo-body"></path>
</svg> </svg>
</a> </a>
<img src="images/logo-full.png" alt="jscanify logo" style="height: 100px" /> <img src="images/logo-full.png" alt="jscanify logo" style="width: 375px" />
<h2>the javascript document scanning library.</h2> <h2>the javascript document scanning library.</h2>
<br /> <br />
<div class="view-on"> <div class="view-on">

3274
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -1,6 +1,6 @@
{ {
"name": "jscanify", "name": "jscanify",
"version": "1.3.2", "version": "1.4.0",
"description": "Open-source Javascript mobile document scanner.", "description": "Open-source Javascript mobile document scanner.",
"main": "src/jscanify-node.js", "main": "src/jscanify-node.js",
"directories": { "directories": {
@ -25,8 +25,8 @@
}, },
"homepage": "https://colonelparrot.github.io/jscanify/", "homepage": "https://colonelparrot.github.io/jscanify/",
"dependencies": { "dependencies": {
"canvas": "^2.11.2", "canvas": "^3.1.0",
"jsdom": "^22.0.0", "jsdom": "^26.0.0",
"mocha": "^10.2.0" "mocha": "^11.1.0"
} }
} }

View File

@ -1,4 +1,4 @@
/*! jscanify v1.3.2 | (c) ColonelParrot and other contributors | MIT License */ /*! jscanify v1.4.0 | (c) ColonelParrot and other contributors | MIT License */
const { Canvas, createCanvas, Image, ImageData } = require("canvas"); const { Canvas, createCanvas, Image, ImageData } = require("canvas");
const { JSDOM } = require("jsdom"); const { JSDOM } = require("jsdom");
@ -80,7 +80,10 @@ class jscanify {
} }
} }
const maxContour = contours.get(maxContourIndex); const maxContour =
maxContourIndex >= 0 ?
contours.get(maxContourIndex) :
null;
imgGray.delete(); imgGray.delete();
imgBlur.delete(); imgBlur.delete();
@ -138,6 +141,9 @@ class jscanify {
/** /**
* Extracts and undistorts the image detected within the frame. * Extracts and undistorts the image detected within the frame.
*
* Returns `null` if no paper is detected.
*
* @param {*} image image to process * @param {*} image image to process
* @param {*} resultWidth desired result paper width * @param {*} resultWidth desired result paper width
* @param {*} resultHeight desired result paper height * @param {*} resultHeight desired result paper height
@ -147,7 +153,12 @@ class jscanify {
extractPaper(image, resultWidth, resultHeight, cornerPoints) { extractPaper(image, resultWidth, resultHeight, cornerPoints) {
const canvas = createCanvas(); const canvas = createCanvas();
const img = cv.imread(image); const img = cv.imread(image);
const maxContour = this.findPaperContour(img); const maxContour = cornerPoints ? null : this.findPaperContour(img);
if(maxContour == null && cornerPoints === undefined){
return null;
}
const { const {
topLeftCorner, topLeftCorner,
topRightCorner, topRightCorner,

View File

@ -1,4 +1,4 @@
/*! jscanify v1.3.2 | (c) ColonelParrot and other contributors | MIT License */ /*! jscanify v1.4.0 | (c) ColonelParrot and other contributors | MIT License */
(function (global, factory) { (function (global, factory) {
typeof exports === "object" && typeof module !== "undefined" typeof exports === "object" && typeof module !== "undefined"
@ -71,7 +71,10 @@
} }
} }
const maxContour = contours.get(maxContourIndex); const maxContour =
maxContourIndex >= 0 ?
contours.get(maxContourIndex) :
null;
imgGray.delete(); imgGray.delete();
imgBlur.delete(); imgBlur.delete();
@ -129,7 +132,10 @@
/** /**
* Extracts and undistorts the image detected within the frame. * Extracts and undistorts the image detected within the frame.
* @param {*} image image to process *
* Returns `null` if no paper is detected.
*
* @param {*} image image to process
* @param {*} resultWidth desired result paper width * @param {*} resultWidth desired result paper width
* @param {*} resultHeight desired result paper height * @param {*} resultHeight desired result paper height
* @param {*} cornerPoints optional custom corner points, in case automatic corner points are incorrect * @param {*} cornerPoints optional custom corner points, in case automatic corner points are incorrect
@ -137,10 +143,12 @@
*/ */
extractPaper(image, resultWidth, resultHeight, cornerPoints) { extractPaper(image, resultWidth, resultHeight, cornerPoints) {
const canvas = document.createElement("canvas"); const canvas = document.createElement("canvas");
const img = cv.imread(image); const img = cv.imread(image);
const maxContour = cornerPoints ? null : this.findPaperContour(img);
const maxContour = this.findPaperContour(img); if(maxContour == null && cornerPoints === undefined){
return null;
}
const { const {
topLeftCorner, topLeftCorner,

View File

@ -16,122 +16,125 @@ const path = require("path");
const OUTPUT_FOLDER = __dirname.replaceAll("\\", "/") + "/output/"; const OUTPUT_FOLDER = __dirname.replaceAll("\\", "/") + "/output/";
const TEST_IMAGE_DIRECTORY = path.join( const TEST_IMAGE_DIRECTORY = path.join(
__dirname, __dirname,
"..", "..",
"docs", "docs",
"images", "images",
"test" "test"
); );
/** /**
* delete previously generated output images * delete previously generated output images
*/ */
function setup() { function setup() {
console.log("=== setting up tests ==="); console.log("=== setting up tests ===");
console.log("Deleting previously generated images"); console.log("Deleting previously generated images");
if (!existsSync(OUTPUT_FOLDER)) { if (!existsSync(OUTPUT_FOLDER)) {
mkdirSync(OUTPUT_FOLDER); mkdirSync(OUTPUT_FOLDER);
} }
readdirSync(OUTPUT_FOLDER).forEach((file) => { readdirSync(OUTPUT_FOLDER).forEach((file) => {
unlinkSync(path.join(OUTPUT_FOLDER, file)); unlinkSync(path.join(OUTPUT_FOLDER, file));
}) })
} }
console.log("=== beginning tests ==="); let scanner;
console.log("loading OpenCV.js..."); let cv;
const scanner = new jscanify(); before(function (done) {
scanner.loadOpenCV(function (cv) { console.log("=== beginning tests ===");
console.log("loading OpenCV.js...");
console.log("Finished loading OpenCV.js"); scanner = new jscanify();
console.log("Writing test images to: " + OUTPUT_FOLDER); scanner.loadOpenCV(function (loadedCv) {
cv = loadedCv;
/** console.log("Finished loading OpenCV.js");
* tests an individual image console.log("Writing test images to: " + OUTPUT_FOLDER);
*/ setup()
function test(testImage, imageCount) { done();
});
});
/**
* tests an individual image
*/
function test(testImage, imageCount) {
describe("image #" + imageCount, function () { describe("image #" + imageCount, function () {
it("should highlight paper", function (done) { it("should highlight paper", function () {
const highlighted = scanner.highlightPaper(testImage); const highlighted = scanner.highlightPaper(testImage);
const higlightedOutputPath = OUTPUT_FOLDER + "highlighted-" + imageCount + ".jpg"; const higlightedOutputPath = OUTPUT_FOLDER + "highlighted-" + imageCount + ".jpg";
writeFileSync( writeFileSync(
higlightedOutputPath, higlightedOutputPath,
highlighted.toBuffer("image/jpeg") highlighted.toBuffer("image/jpeg")
); );
assert.ok(existsSync(higlightedOutputPath)); assert.ok(existsSync(higlightedOutputPath));
done();
});
it("should extract paper", function (done) {
const extracted = scanner.extractPaper(testImage, 386, 500);
const extractedOutputPath = OUTPUT_FOLDER + "extracted-" + imageCount + ".jpg";
writeFileSync(
extractedOutputPath,
extracted.toBuffer("image/jpeg")
);
assert.ok(existsSync(extractedOutputPath));
done();
});
it("should label corner points", function (done) {
const parsedImage = cv.imread(testImage);
const paperContour = scanner.findPaperContour(parsedImage);
const {
topLeftCorner,
topRightCorner,
bottomLeftCorner,
bottomRightCorner,
} = scanner.getCornerPoints(paperContour, testImage);
const canvas = createCanvas();
cv.imshow(canvas, parsedImage);
const ctx = canvas.getContext("2d");
const points = [
{ p: topLeftCorner, text: "top left corner" },
{ p: topRightCorner, text: "top right corner" },
{ p: bottomLeftCorner, text: "bottom left corner" },
{ p: bottomRightCorner, text: "bottom right corner" },
];
ctx.fillStyle = "cyan";
ctx.font = "25px serif";
points.forEach(({ p: point, text }) => {
ctx.beginPath();
ctx.arc(point.x, point.y, 15, 0, 2 * Math.PI, false);
ctx.fillText(text, point.x + 30, point.y)
ctx.fill();
}); });
const cornerPointsOutputPath = OUTPUT_FOLDER + "corner_points-" + imageCount + ".jpg"; it("should extract paper", function () {
writeFileSync(cornerPointsOutputPath, canvas.toBuffer("image/jpeg")); const extracted = scanner.extractPaper(testImage, 386, 500);
const extractedOutputPath = OUTPUT_FOLDER + "extracted-" + imageCount + ".jpg";
assert.ok(existsSync(cornerPointsOutputPath)); writeFileSync(
done(); extractedOutputPath,
}); extracted.toBuffer("image/jpeg")
);
assert.ok(existsSync(extractedOutputPath));
});
it("should label corner points", function () {
const parsedImage = cv.imread(testImage);
const paperContour = scanner.findPaperContour(parsedImage);
const {
topLeftCorner,
topRightCorner,
bottomLeftCorner,
bottomRightCorner,
} = scanner.getCornerPoints(paperContour, testImage);
const canvas = createCanvas();
cv.imshow(canvas, parsedImage);
const ctx = canvas.getContext("2d");
const points = [
{ p: topLeftCorner, text: "top left corner" },
{ p: topRightCorner, text: "top right corner" },
{ p: bottomLeftCorner, text: "bottom left corner" },
{ p: bottomRightCorner, text: "bottom right corner" },
];
ctx.fillStyle = "cyan";
ctx.font = "25px serif";
points.forEach(({ p: point, text }) => {
ctx.beginPath();
ctx.arc(point.x, point.y, 15, 0, 2 * Math.PI, false);
ctx.fillText(text, point.x + 30, point.y)
ctx.fill();
});
const cornerPointsOutputPath = OUTPUT_FOLDER + "corner_points-" + imageCount + ".jpg";
writeFileSync(cornerPointsOutputPath, canvas.toBuffer("image/jpeg"));
assert.ok(existsSync(cornerPointsOutputPath));
});
}); });
} }
setup(); let imageCount = 1;
let imageCount = 1; /*
* go through all images in test image directory
/* */
* go through all images in test image directory readdirSync(TEST_IMAGE_DIRECTORY).forEach((file) => {
*/
readdirSync(TEST_IMAGE_DIRECTORY).forEach((file) => {
const TEST_IMAGE_PATH = path.join(TEST_IMAGE_DIRECTORY, file); const TEST_IMAGE_PATH = path.join(TEST_IMAGE_DIRECTORY, file);
if(!file.endsWith("-sized.png")){ // these images are for the website, not testing if (!file.endsWith("-sized.png")) { // these images are for the website, not testing
let tempCount = imageCount++; let tempCount = imageCount++;
loadImage(TEST_IMAGE_PATH).then(function (image) { loadImage(TEST_IMAGE_PATH).then(function (image) {
test(image, tempCount); test(image, tempCount);
}); });
} }
}) })
});