/* * main.cpp * zxing * * Created by Ralf Kistner on 16/10/2009. * Copyright 2008 ZXing authors All rights reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include #include #include #include #include "MagickBitmapSource.h" #include #include #include #include #include #include #include #include #include #include #include #include using namespace Magick; using namespace std; using namespace zxing; using namespace zxing::qrcode; void draw_matrix(Image& image, Ref matrix) { int width = matrix->getWidth(); int height = matrix->getHeight(); // image.modifyImage(); // image.type(TrueColorType); PixelPacket* pixels = image.getPixels(0, 0, width, height); PixelPacket* pixel = pixels; ColorMono color; for (int y = 0; y < height; y++) { for (int x = 0; x < width; x++) { color.mono(!matrix->get(x, y)); *pixel = color; pixel++; } } image.syncPixels(); } void save_matrix(Ref matrix, string filename, float scale = 1.0) { Image image(Geometry(matrix->getWidth(), matrix->getHeight()), Color(MaxRGB, MaxRGB, MaxRGB, 0)); int width = matrix->getWidth(); int height = matrix->getHeight(); draw_matrix(image, matrix); image.scale(Geometry(width*scale, height*scale)); image.write(filename); } void save_grid(Ref matrix, string filename, Ref transform, int dimension) { Image image(Geometry(matrix->getWidth(), matrix->getHeight()), Color(MaxRGB, MaxRGB, MaxRGB, 0)); draw_matrix(image, matrix); image.strokeColor(Color(MaxRGB, 0, 0, MaxRGB / 3)); image.fillColor(Color(0, 0, 0, MaxRGB)); image.strokeWidth(1); for (int i = 0; i <= dimension; i++) { valarray tpoints(0.0, 4); tpoints[0] = 0; tpoints[1] = i; tpoints[2] = dimension; tpoints[3] = i; transform->transformPoints(tpoints); DrawableLine line1(tpoints[0], tpoints[1], tpoints[2], tpoints[3]); image.draw(line1); tpoints[0] = i; tpoints[1] = 0; tpoints[2] = i; tpoints[3] = dimension; transform->transformPoints(tpoints); DrawableLine line2(tpoints[0], tpoints[1], tpoints[2], tpoints[3]); image.draw(line2); } image.write(filename); } Ref decode(string out_prefix, Ref image, string& cell_grid, string& cell_transformed) { Decoder decoder; Detector detector(image->getBlackMatrix()); Ref detectorResult(detector.detect()); if (out_prefix.size()) { // Grid image string gridfile = out_prefix + ".grid.gif"; Ref transform = detectorResult->getTransform(); int dimension = detectorResult->getBits()->getDimension(); save_grid(image->getBlackMatrix(), gridfile, transform, dimension); cell_grid = ""; // Transformed image string tfile = out_prefix + ".transformed.png"; save_matrix(detectorResult->getBits(), tfile, 5); cell_transformed = ""; } vector > points(detectorResult->getPoints()); Ref decoderResult(decoder.decode(detectorResult->getBits())); Ref result(new Result(decoderResult->getText(), decoderResult->getRawBytes(), points, BarcodeFormat_QR_CODE)); return result; } int test_image(Image& image, string out_prefix, bool localized) { string cell_mono; string cell_transformed; string cell_result; string cell_grid; int res = -1; Ref matrix(NULL); Ref binarizer(NULL); try { Ref source(new MagickBitmapSource(image)); if (localized) { binarizer = new LocalBlockBinarizer(source); } else { binarizer = new GlobalHistogramBinarizer(source); } if (out_prefix.size()) { string monofile = out_prefix + ".mono.png"; matrix = binarizer->getBlackMatrix(); save_matrix(matrix, monofile); cell_mono = ""; } Ref binary(new BinaryBitmap(binarizer)); Ref result(decode(out_prefix, binary, cell_grid, cell_transformed)); cell_result = "" + result->getText()->getText() + ""; res = 0; } catch (ReaderException e) { cell_result = "zxing::ReaderException: " + string(e.what()); res = -2; } catch (zxing::IllegalArgumentException& e) { cell_result = "zxing::IllegalArgumentException: " + string(e.what()); res = -3; } catch (zxing::Exception& e) { cell_result = "zxing::Exception: " + string(e.what()); res = -4; } catch (std::exception& e) { cell_result = "std::exception: " + string(e.what()); res = -5; } cout << "" << cell_mono << "" << endl; cout << "" << cell_grid << "" << endl; cout << "" << cell_transformed << "" << endl; cout << "" << cell_result << "" << endl; return res; } int test_image_local(Image& image, string out_prefix) { return test_image(image, out_prefix, true); } int test_image_global(Image& image, string out_prefix) { return test_image(image, out_prefix, false); } int main(int argc, char** argv) { if (argc <= 2) { cout << "Usage: " << argv[0] << " [ | \"-\"] [ ...]" << endl; return 1; } string outfolder = argv[1]; int total = argc - 2; int gonly = 0; int lonly = 0; int both = 0; int neither = 0; cout << "" << endl; for (int i = 2; i < argc; i++) { string infilename = argv[i]; cerr << "Processing: " << infilename << endl; Image image; try { image.read(infilename); } catch (...) { cerr << "Unable to open image, ignoring" << endl; continue; } cout << "" << endl; cout << "" << endl; cout << "" << endl; int gresult = 1; int lresult = 1; if (outfolder == string("-")) { gresult = test_image_global(image, ""); lresult = test_image_local(image, ""); } else { replace(infilename.begin(), infilename.end(), '/', '_'); string prefix = string(outfolder) + string("/") + infilename; gresult = test_image_global(image, prefix + ".g"); lresult = test_image_local(image, prefix + ".l"); } gresult = gresult == 0; lresult = lresult == 0; gonly += gresult && !lresult; lonly += lresult && !gresult; both += gresult && lresult; neither += !gresult && !lresult; cout << "" << endl; } cout << "
" << infilename << "
" << endl; cout << "" << endl; cout << "" << endl; cout << "" << endl; cout << "" << endl; cout << "" << endl; cout << "" << endl; cout << "
Total" << total << "
Both correct" << both << "
Neither correct" << neither << "
Global only" << gonly << "
Local only" << lonly << "
" << endl; cout << "" << endl; return 0; }