diff --git a/Capstone_Design.sln b/Capstone_Design.sln
index 2c7f033..758e61b 100644
--- a/Capstone_Design.sln
+++ b/Capstone_Design.sln
@@ -65,6 +65,20 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "edge", "Homework\edge\edge.
EndProject
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "YCbCrHist", "Homework\YCbCrHist\YCbCrHist.vcxproj", "{2DB82859-6BEC-40B4-A8CA-BFA2C60145B0}"
EndProject
+Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "ch11", "ch11", "{94FE4669-77AC-4D41-8E24-F5DEDEAC930F}"
+EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "thershhold", "ch11\thershhold\thershhold.vcxproj", "{12A25565-0007-412B-9490-9AA28305272D}"
+EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "adaptive", "ch11\adaptive\adaptive.vcxproj", "{8F873187-A5F8-4D02-ABE9-BF652FE53BC3}"
+EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "morphology", "ch11\morphology\morphology.vcxproj", "{9BD9A147-29ED-4F6C-8BFD-71F03E56332C}"
+EndProject
+Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "ch12", "ch12", "{792AFFC9-6147-4A66-B676-D64FB3E62F2E}"
+EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "labeling", "ch12\labeling\labeling.vcxproj", "{AC2741E6-3BF6-417D-8CE5-82696760AC35}"
+EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "findcts", "ch12\findcts\findcts.vcxproj", "{6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|x64 = Debug|x64
@@ -241,6 +255,46 @@ Global
{2DB82859-6BEC-40B4-A8CA-BFA2C60145B0}.Release|x64.Build.0 = Release|x64
{2DB82859-6BEC-40B4-A8CA-BFA2C60145B0}.Release|x86.ActiveCfg = Release|Win32
{2DB82859-6BEC-40B4-A8CA-BFA2C60145B0}.Release|x86.Build.0 = Release|Win32
+ {12A25565-0007-412B-9490-9AA28305272D}.Debug|x64.ActiveCfg = Debug|x64
+ {12A25565-0007-412B-9490-9AA28305272D}.Debug|x64.Build.0 = Debug|x64
+ {12A25565-0007-412B-9490-9AA28305272D}.Debug|x86.ActiveCfg = Debug|Win32
+ {12A25565-0007-412B-9490-9AA28305272D}.Debug|x86.Build.0 = Debug|Win32
+ {12A25565-0007-412B-9490-9AA28305272D}.Release|x64.ActiveCfg = Release|x64
+ {12A25565-0007-412B-9490-9AA28305272D}.Release|x64.Build.0 = Release|x64
+ {12A25565-0007-412B-9490-9AA28305272D}.Release|x86.ActiveCfg = Release|Win32
+ {12A25565-0007-412B-9490-9AA28305272D}.Release|x86.Build.0 = Release|Win32
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Debug|x64.ActiveCfg = Debug|x64
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Debug|x64.Build.0 = Debug|x64
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Debug|x86.ActiveCfg = Debug|Win32
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Debug|x86.Build.0 = Debug|Win32
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Release|x64.ActiveCfg = Release|x64
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Release|x64.Build.0 = Release|x64
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Release|x86.ActiveCfg = Release|Win32
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3}.Release|x86.Build.0 = Release|Win32
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Debug|x64.ActiveCfg = Debug|x64
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Debug|x64.Build.0 = Debug|x64
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Debug|x86.ActiveCfg = Debug|Win32
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Debug|x86.Build.0 = Debug|Win32
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Release|x64.ActiveCfg = Release|x64
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Release|x64.Build.0 = Release|x64
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Release|x86.ActiveCfg = Release|Win32
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C}.Release|x86.Build.0 = Release|Win32
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Debug|x64.ActiveCfg = Debug|x64
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Debug|x64.Build.0 = Debug|x64
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Debug|x86.ActiveCfg = Debug|Win32
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Debug|x86.Build.0 = Debug|Win32
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Release|x64.ActiveCfg = Release|x64
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Release|x64.Build.0 = Release|x64
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Release|x86.ActiveCfg = Release|Win32
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35}.Release|x86.Build.0 = Release|Win32
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Debug|x64.ActiveCfg = Debug|x64
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Debug|x64.Build.0 = Debug|x64
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Debug|x86.ActiveCfg = Debug|Win32
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Debug|x86.Build.0 = Debug|Win32
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Release|x64.ActiveCfg = Release|x64
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Release|x64.Build.0 = Release|x64
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Release|x86.ActiveCfg = Release|Win32
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E}.Release|x86.Build.0 = Release|Win32
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
@@ -267,6 +321,11 @@ Global
{F7126457-CC07-433A-B762-C4DFB352653C} = {2BBACE58-DAC1-4D1E-85FC-3A3E9F789D9B}
{85DBF128-C338-4C43-B575-87C1B8CD2EFA} = {E583E257-9FCE-4A4B-A554-8514F982D1BF}
{2DB82859-6BEC-40B4-A8CA-BFA2C60145B0} = {E583E257-9FCE-4A4B-A554-8514F982D1BF}
+ {12A25565-0007-412B-9490-9AA28305272D} = {94FE4669-77AC-4D41-8E24-F5DEDEAC930F}
+ {8F873187-A5F8-4D02-ABE9-BF652FE53BC3} = {94FE4669-77AC-4D41-8E24-F5DEDEAC930F}
+ {9BD9A147-29ED-4F6C-8BFD-71F03E56332C} = {94FE4669-77AC-4D41-8E24-F5DEDEAC930F}
+ {AC2741E6-3BF6-417D-8CE5-82696760AC35} = {792AFFC9-6147-4A66-B676-D64FB3E62F2E}
+ {6F3AAF9A-D050-4BE9-A2F7-3E0CAB16E69E} = {792AFFC9-6147-4A66-B676-D64FB3E62F2E}
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {E1A0C3F4-F16D-4626-86CE-D8484C0C776D}
diff --git a/ch11/adaptive/adaptive.vcxproj b/ch11/adaptive/adaptive.vcxproj
new file mode 100644
index 0000000..7f139fd
--- /dev/null
+++ b/ch11/adaptive/adaptive.vcxproj
@@ -0,0 +1,139 @@
+
+
+
+
+ Debug
+ Win32
+
+
+ Release
+ Win32
+
+
+ Debug
+ x64
+
+
+ Release
+ x64
+
+
+
+ 17.0
+ Win32Proj
+ {8f873187-a5f8-4d02-abe9-bf652fe53bc3}
+ adaptive
+ 10.0
+
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Level3
+ true
+ WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+ Level3
+ true
+ _DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/ch11/adaptive/adaptive.vcxproj.filters b/ch11/adaptive/adaptive.vcxproj.filters
new file mode 100644
index 0000000..2c1b3dc
--- /dev/null
+++ b/ch11/adaptive/adaptive.vcxproj.filters
@@ -0,0 +1,22 @@
+
+
+
+
+ {4FC737F1-C7A5-4376-A066-2A32D752A2FF}
+ cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx
+
+
+ {93995380-89BD-4b04-88EB-625FBE52EBFB}
+ h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd
+
+
+ {67DA6AB6-F800-4c08-8B7A-83BB121AAD01}
+ rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms
+
+
+
+
+ 소스 파일
+
+
+
\ No newline at end of file
diff --git a/ch11/adaptive/main.cpp b/ch11/adaptive/main.cpp
new file mode 100644
index 0000000..b33a6ac
--- /dev/null
+++ b/ch11/adaptive/main.cpp
@@ -0,0 +1,43 @@
+#include
+#include "opencv2/opencv.hpp"
+
+using namespace cv;
+using namespace std;
+
+void on_trackbar(int pos, void* userdata);
+
+int main(int argc, char* argv[]) {
+ Mat src;
+
+ if (argc < 2)
+ src = imread("../../resources/images/sudoku.jpg", IMREAD_GRAYSCALE);
+ else
+ src = imread(argv[1], IMREAD_GRAYSCALE);
+
+ if (src.empty()) {
+ cerr << "Image load failed!" << endl;
+ return -1;
+ }
+
+ imshow("src", src);
+
+ namedWindow("dst");
+ createTrackbar("Block size", "dst", 0, 200, on_trackbar, (void*)&src);
+ setTrackbarPos("Block size", "dst", 11);
+
+ cv::waitKey();
+ cv::destroyAllWindows();
+}
+
+void on_trackbar(int pos, void* userdata) {
+ Mat src = *(Mat*)userdata;
+
+ int bsize = pos;
+ if (bsize % 2 == 0) bsize--;
+ if (bsize < 3) bsize = 3;
+
+ Mat dst;
+ adaptiveThreshold(src, dst, 255, ADAPTIVE_THRESH_GAUSSIAN_C, THRESH_BINARY, bsize, 5);
+
+ imshow("dst", dst);
+}
\ No newline at end of file
diff --git a/ch11/morphology/main.cpp b/ch11/morphology/main.cpp
new file mode 100644
index 0000000..5e7bdf8
--- /dev/null
+++ b/ch11/morphology/main.cpp
@@ -0,0 +1,72 @@
+#include
+#include "opencv2/opencv.hpp"
+
+using namespace cv;
+using namespace std;
+
+void erode_dilate() {
+ Mat src = imread("../../resources/images/milkdrop.bmp", IMREAD_GRAYSCALE);
+ Mat bin;
+ threshold(src, bin, 0, 255, THRESH_BINARY | THRESH_OTSU);
+
+ Mat dst1, dst2;
+ erode(bin, dst1, getStructuringElement(MorphShapes::MORPH_CROSS, Size(7, 7)));
+ dilate(bin, dst2, getStructuringElement(MorphShapes::MORPH_ELLIPSE, Size(3, 5)));
+
+ imshow("src", src);
+ imshow("bin", bin);
+ imshow("erode", dst1);
+ imshow("dilate", dst2);
+
+ cv::waitKey();
+ cv::destroyAllWindows();
+}
+
+void open_close() {
+ cv::VideoCapture video("../../resources/images/vtest.avi");
+ if (!video.isOpened()) {
+ std::cerr << "camera.isopened()" << std::endl;
+ return;
+ }
+ cv::Mat src;
+ cv::Mat prevSrc;
+ double fps = 1000 / video.get(cv::VideoCaptureProperties::CAP_PROP_FPS);
+ std::cout << "\n\n" << fps << "\n\n";
+ while (true) {
+ prevSrc = src;
+ video >> src;
+ if (src.empty()) {
+ std::cerr << "img.empty()" << std::endl;
+ return;
+ }
+ cv::imshow("src", src);
+ if (cv::waitKey(fps) == 27)
+ break;
+ }
+ cv::cvtColor(src, src, cv::COLOR_BGR2GRAY);
+ src.convertTo(src, CV_8U);
+
+ Mat bin;
+ threshold(src, bin, 0, 255, THRESH_BINARY | THRESH_OTSU);
+
+ Mat dst1, dst2, dst3, dst4, dst5;
+ morphologyEx(bin, dst1, MORPH_OPEN, Mat());
+ morphologyEx(bin, dst2, MORPH_CLOSE, Mat());
+ morphologyEx(bin, dst3, MORPH_GRADIENT, Mat());
+ Canny(bin, dst4, 30, 127);
+ dst5 = std::abs(src - prevSrc);
+
+ imshow("src", src);
+ imshow("bin", bin);
+ imshow("erode", dst1);
+ imshow("dilate", dst2);
+ imshow("gradient", dst3);
+ imshow("Canny", dst4);
+
+ cv::waitKey();
+ cv::destroyAllWindows();
+}
+
+int main(int argc, char* argv[]) {
+ open_close();
+}
\ No newline at end of file
diff --git a/ch11/morphology/morphology.vcxproj b/ch11/morphology/morphology.vcxproj
new file mode 100644
index 0000000..16233c7
--- /dev/null
+++ b/ch11/morphology/morphology.vcxproj
@@ -0,0 +1,145 @@
+
+
+
+
+ Debug
+ Win32
+
+
+ Release
+ Win32
+
+
+ Debug
+ x64
+
+
+ Release
+ x64
+
+
+
+ 17.0
+ Win32Proj
+ {9bd9a147-29ed-4f6c-8bfd-71f03e56332c}
+ morphology
+ 10.0
+
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Level3
+ true
+ WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+ Level3
+ true
+ _DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/ch11/morphology/morphology.vcxproj.filters b/ch11/morphology/morphology.vcxproj.filters
new file mode 100644
index 0000000..f0f8d4f
--- /dev/null
+++ b/ch11/morphology/morphology.vcxproj.filters
@@ -0,0 +1,32 @@
+
+
+
+
+ {4FC737F1-C7A5-4376-A066-2A32D752A2FF}
+ cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx
+
+
+ {93995380-89BD-4b04-88EB-625FBE52EBFB}
+ h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd
+
+
+ {67DA6AB6-F800-4c08-8B7A-83BB121AAD01}
+ rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms
+
+
+
+
+ 소스 파일
+
+
+
+
+ 리소스 파일
+
+
+
+
+ 리소스 파일
+
+
+
\ No newline at end of file
diff --git a/ch11/thershhold/main.cpp b/ch11/thershhold/main.cpp
new file mode 100644
index 0000000..b436cf4
--- /dev/null
+++ b/ch11/thershhold/main.cpp
@@ -0,0 +1,54 @@
+#include
+#include "opencv2/opencv.hpp"
+
+using namespace cv;
+using namespace std;
+
+void on_trackbar(int pos, void* userdata);
+
+int main(int argc, char* argv[]) {
+ Mat src;
+
+ if (argc < 2)
+ src = imread("../../resources/images/neutrophils.png", IMREAD_GRAYSCALE);
+ else
+ src = imread(argv[1], IMREAD_GRAYSCALE);
+
+ if (src.empty()) {
+ cerr << "Image load failed!" << endl;
+ return -1;
+ }
+
+ namedWindow("dst");
+ createTrackbar("Threshold", "dst", 0, 255, on_trackbar, (void*)&src);
+ setTrackbarPos("Threshold", "dst", 128);
+
+ cv::waitKey();
+ cv::destroyAllWindows();
+}
+
+void on_trackbar(int pos, void* userdata) {
+ Mat src = *(Mat*)userdata;
+
+ Mat hist;
+ int channels[] = { 0 };
+ int dims = 1;
+ const int histSize[] = { 256 };
+ float graylevel[] = { 0, 256 };
+ const float* ranges[] = { graylevel };
+
+ calcHist(&src, 1, channels, noArray(), hist, dims, histSize, ranges);
+ double histMax;
+ minMaxLoc(hist, 0, &histMax);
+ Mat imgHist(100, 256, CV_8UC1, Scalar(255));
+ for (int i = 0; i < 256; i++) {
+ line(imgHist, Point(i, 100), Point(i, 100 - cvRound(hist.at(i, 0) * 100 / histMax)), Scalar(0));
+ }
+ line(imgHist, Point(pos, 0), Point(pos, 100), 0, 1, LINE_AA);
+
+ Mat dst;
+ threshold(src, dst, pos, 255, THRESH_BINARY);
+
+ imshow("dst", dst);
+ imshow("hist", imgHist);
+}
\ No newline at end of file
diff --git a/ch11/thershhold/thershhold.vcxproj b/ch11/thershhold/thershhold.vcxproj
new file mode 100644
index 0000000..902a65b
--- /dev/null
+++ b/ch11/thershhold/thershhold.vcxproj
@@ -0,0 +1,144 @@
+
+
+
+
+ Debug
+ Win32
+
+
+ Release
+ Win32
+
+
+ Debug
+ x64
+
+
+ Release
+ x64
+
+
+
+ 17.0
+ Win32Proj
+ {12a25565-0007-412b-9490-9aa28305272d}
+ thershhold
+ 10.0
+
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Level3
+ true
+ WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+ Level3
+ true
+ _DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/ch11/thershhold/thershhold.vcxproj.filters b/ch11/thershhold/thershhold.vcxproj.filters
new file mode 100644
index 0000000..813d112
--- /dev/null
+++ b/ch11/thershhold/thershhold.vcxproj.filters
@@ -0,0 +1,33 @@
+
+
+
+
+ {4FC737F1-C7A5-4376-A066-2A32D752A2FF}
+ cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx
+
+
+ {93995380-89BD-4b04-88EB-625FBE52EBFB}
+ h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd
+
+
+ {67DA6AB6-F800-4c08-8B7A-83BB121AAD01}
+ rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms
+
+
+
+
+ 소스 파일
+
+
+
+
+ 리소스 파일
+
+
+ 리소스 파일
+
+
+ 리소스 파일
+
+
+
\ No newline at end of file
diff --git a/ch12/findcts/findcts.vcxproj b/ch12/findcts/findcts.vcxproj
new file mode 100644
index 0000000..970af53
--- /dev/null
+++ b/ch12/findcts/findcts.vcxproj
@@ -0,0 +1,142 @@
+
+
+
+
+ Debug
+ Win32
+
+
+ Release
+ Win32
+
+
+ Debug
+ x64
+
+
+ Release
+ x64
+
+
+
+ 17.0
+ Win32Proj
+ {6f3aaf9a-d050-4be9-a2f7-3e0cab16e69e}
+ findcts
+ 10.0
+
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Level3
+ true
+ WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+ Level3
+ true
+ _DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/ch12/findcts/findcts.vcxproj.filters b/ch12/findcts/findcts.vcxproj.filters
new file mode 100644
index 0000000..05f0d88
--- /dev/null
+++ b/ch12/findcts/findcts.vcxproj.filters
@@ -0,0 +1,27 @@
+
+
+
+
+ {4FC737F1-C7A5-4376-A066-2A32D752A2FF}
+ cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx
+
+
+ {93995380-89BD-4b04-88EB-625FBE52EBFB}
+ h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd
+
+
+ {67DA6AB6-F800-4c08-8B7A-83BB121AAD01}
+ rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms
+
+
+
+
+ 소스 파일
+
+
+
+
+ 리소스 파일
+
+
+
\ No newline at end of file
diff --git a/ch12/findcts/main.cpp b/ch12/findcts/main.cpp
new file mode 100644
index 0000000..9fc2e91
--- /dev/null
+++ b/ch12/findcts/main.cpp
@@ -0,0 +1,133 @@
+#include
+#include "opencv2/opencv.hpp"
+using namespace cv;
+
+void contours_basic() {
+ Mat src = imread("../../resources/images/contours.bmp", IMREAD_GRAYSCALE);
+
+ std::vector> contours;
+ findContours(src, contours, RETR_LIST, CHAIN_APPROX_NONE);
+
+ Mat dst;
+ cvtColor(src, dst, COLOR_GRAY2BGR);
+
+ for (int i = 0; i < contours.size(); i++) {
+ Scalar c(rand() & 255, rand() & 255, rand() & 255);
+ drawContours(dst, contours, i, c, 2);
+ }
+
+ imshow("src", src);
+ imshow("dst", dst);
+
+ waitKey();
+ destroyAllWindows();
+}
+
+void contours_hier() {
+ Mat src = imread("../../resources/images/contours.bmp", IMREAD_GRAYSCALE);
+
+ std::vector> contours;
+ std::vector hirarchy;
+ findContours(src, contours, hirarchy, RETR_CCOMP, CHAIN_APPROX_NONE);
+
+ Mat dst;
+ cvtColor(src, dst, COLOR_GRAY2BGR);
+
+ for (int i = 0; i >= 0; i = hirarchy[i][0]) {
+ Scalar c(rand() & 255, rand() & 255, rand() & 255);
+ drawContours(dst, contours, i, c, -1, LINE_8, hirarchy);
+ }
+
+ for (int i = 0; i < contours.size(); i++) {
+ std::cout << "Hirarchy of contour index: " << i << "-->" << hirarchy[i] << std::endl;
+ }
+
+ imshow("src", src);
+ imshow("dst", dst);
+
+ waitKey();
+ destroyAllWindows();
+}
+
+void contour_apps() {
+ Mat src = imread("../../resources/images/beta.png", IMREAD_GRAYSCALE);
+
+ Mat bin;
+ std::vector> contours;
+ std::vector hirarchy;
+ threshold(src, bin, 128, 255, THRESH_BINARY | THRESH_OTSU);
+ morphologyEx(bin, bin, MORPH_OPEN, Mat(), Point(-1, -1), 1);
+
+ findContours(bin, contours, hirarchy, RETR_TREE, CHAIN_APPROX_NONE);
+
+ Mat dst;
+ cvtColor(src, dst, COLOR_GRAY2BGR);
+
+ for (int idx = 0; idx >= 0; idx = hirarchy[idx][0]) {
+ Scalar c(rand() & 255, rand() & 255, rand() & 255);
+ drawContours(dst, contours, idx, c, 2, LINE_AA, hirarchy, 2, Point(0, 0));
+
+ Rect rect = boundingRect(contours[idx]);
+ rectangle(dst, rect, Scalar(0, 0, 255), 2, LINE_AA, 0);
+
+ RotatedRect rrect = minAreaRect(contours[idx]);
+ Point2f vertices[4];
+ rrect.points(vertices);
+ for (int i = 0; i < 4; i++)
+ line(dst, vertices[i], vertices[(i + 1 & 3)], Scalar(255, 0, 0), 2, LINE_AA, 0);
+ float angle = rrect.angle;
+ std::cout << "\nrotated angle: " << cvRound(angle) << std::endl;
+
+ Point2f center;
+ float radius;
+ minEnclosingCircle(contours[idx], center, radius);
+ circle(dst, center, radius, Scalar(0, 255, 255), 2, LINE_AA, 0);
+
+ double len = arcLength(contours[idx], true);
+ std::cout << "\nlength of contour: " << cvRound(len) << std::endl;
+
+ double area = contourArea(contours[idx], false);
+ std::cout << "\narea of contour: " << cvRound(area) << std::endl;
+ }
+
+ imshow("dst", dst);
+
+ waitKey();
+ destroyAllWindows();
+}
+
+void contour_approximation() {
+ Mat src = imread("../../resources/images/kmap_simple.jpg", IMREAD_GRAYSCALE);
+
+ Mat bin;
+ std::vector> contours;
+ std::vector hirarchy;
+ threshold(src, bin, 128, 255, THRESH_BINARY | THRESH_OTSU);
+ findContours(bin, contours, hirarchy, RETR_TREE, CHAIN_APPROX_NONE);
+ Mat dst;
+ cvtColor(src, dst, COLOR_GRAY2BGR);
+ for (int idx = 0; idx >= 0; idx = hirarchy[idx][0]) {
+ Scalar c(rand() & 255, rand() & 255, rand() & 255);
+ drawContours(dst, contours, idx, c, 2, LINE_AA, hirarchy, 2, Point(0, 0));
+
+ std::vector approxCurve;
+ approxPolyDP(contours[idx], approxCurve, arcLength(contours[idx], true) * 0.005, true);
+ int no_vertex = approxCurve.size();
+ std::cout << "The nimber of vertex: " << no_vertex << std::endl;
+ for (int v = 0; v < no_vertex; v++) {
+ circle(dst, approxCurve[v], 3, Scalar(0, 0, 255), 2);
+ line(dst, approxCurve[v], approxCurve[(v+1) % no_vertex], Scalar(255, 255, 0), 2);
+ }
+
+ }
+
+ imshow("src", src);
+ imshow("dst", dst);
+
+ waitKey();
+ destroyAllWindows();
+}
+
+int main() {
+ contour_approximation();
+}
\ No newline at end of file
diff --git a/ch12/labeling/labeling.vcxproj b/ch12/labeling/labeling.vcxproj
new file mode 100644
index 0000000..3890137
--- /dev/null
+++ b/ch12/labeling/labeling.vcxproj
@@ -0,0 +1,142 @@
+
+
+
+
+ Debug
+ Win32
+
+
+ Release
+ Win32
+
+
+ Debug
+ x64
+
+
+ Release
+ x64
+
+
+
+ 17.0
+ Win32Proj
+ {ac2741e6-3bf6-417d-8ce5-82696760ac35}
+ labeling
+ 10.0
+
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+ Application
+ true
+ v143
+ Unicode
+
+
+ Application
+ false
+ v143
+ true
+ Unicode
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Level3
+ true
+ WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+ Level3
+ true
+ _DEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+
+
+
+
+ Level3
+ true
+ true
+ true
+ NDEBUG;_CONSOLE;%(PreprocessorDefinitions)
+ true
+
+
+ Console
+ true
+ true
+ true
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/ch12/labeling/labeling.vcxproj.filters b/ch12/labeling/labeling.vcxproj.filters
new file mode 100644
index 0000000..2def071
--- /dev/null
+++ b/ch12/labeling/labeling.vcxproj.filters
@@ -0,0 +1,27 @@
+
+
+
+
+ {4FC737F1-C7A5-4376-A066-2A32D752A2FF}
+ cpp;c;cc;cxx;c++;cppm;ixx;def;odl;idl;hpj;bat;asm;asmx
+
+
+ {93995380-89BD-4b04-88EB-625FBE52EBFB}
+ h;hh;hpp;hxx;h++;hm;inl;inc;ipp;xsd
+
+
+ {67DA6AB6-F800-4c08-8B7A-83BB121AAD01}
+ rc;ico;cur;bmp;dlg;rc2;rct;bin;rgs;gif;jpg;jpeg;jpe;resx;tiff;tif;png;wav;mfcribbon-ms
+
+
+
+
+ 소스 파일
+
+
+
+
+ 리소스 파일
+
+
+
\ No newline at end of file
diff --git a/ch12/labeling/main.cpp b/ch12/labeling/main.cpp
new file mode 100644
index 0000000..b9b82f1
--- /dev/null
+++ b/ch12/labeling/main.cpp
@@ -0,0 +1,61 @@
+#include
+#include "opencv2/opencv.hpp"
+using namespace cv;
+
+void labeling_basic() {
+ uchar data[] = {
+ 0, 0, 1, 1, 0, 0, 0, 0,
+ 1, 1, 1, 1, 0, 0, 1, 0,
+ 1, 1, 1, 1, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 1, 1, 0,
+ 0, 0, 0, 1, 1, 1, 1, 0,
+ 0, 0, 0, 1, 0, 0, 1, 0,
+ 0, 0, 1, 1, 1, 1, 1, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0
+ };
+
+ Mat src = Mat(8, 8, CV_8UC1, data) * 255;
+
+ Mat labels;
+ int cnt = connectedComponents(src, labels);
+
+ std::cout << "src:\n" << src << std::endl;
+ std::cout << "lables:\n" << labels << std::endl;
+ std::cout << "number of lables:\n" << cnt << std::endl;
+}
+
+void labling_stats() {
+ Mat src = imread("../../resources/images/keyboard.bmp", IMREAD_GRAYSCALE);
+
+ Mat bin;
+ threshold(src, bin, 0, 255, THRESH_BINARY | THRESH_OTSU);
+
+ Mat labels, stats, centroids;
+ int cnt = connectedComponentsWithStats(bin, labels, stats, centroids);
+
+ Mat dst;
+ cvtColor(src, dst, COLOR_GRAY2BGR);
+
+ for (int i = 1; i < cnt; i++) {
+ int* p = stats.ptr(i);
+ double* c = centroids.ptr(i);
+
+ //if (p[4] < 20)
+ // continue;
+
+ rectangle(dst, Rect(p[0], p[1], p[2], p[3]), Scalar(0, 255, 255), 2);
+ char num[30] = { '0' + i, '\0'};
+ sprintf_s(num, "%d", i);
+ putText(dst, num, Point(c[0], c[1]), FONT_HERSHEY_PLAIN, 1.2, Scalar(0, 0, 255));
+ }
+
+ imshow("src", src);
+ imshow("dst", dst);
+
+ waitKey();
+ destroyAllWindows();
+}
+
+int main() {
+ labling_stats();
+}
\ No newline at end of file
diff --git a/resources/images/beta.png b/resources/images/beta.png
new file mode 100644
index 0000000..6e38f9e
--- /dev/null
+++ b/resources/images/beta.png
@@ -0,0 +1,3 @@
+version https://git-lfs.github.com/spec/v1
+oid sha256:f0faf65eec69428ff1683be48488c13a62f0f4579cbe25d37c0091854fb1912f
+size 15725
diff --git a/resources/images/document.bmp b/resources/images/document.bmp
new file mode 100644
index 0000000..553629c
--- /dev/null
+++ b/resources/images/document.bmp
@@ -0,0 +1,3 @@
+version https://git-lfs.github.com/spec/v1
+oid sha256:543e2b7228821792610d3cbe86077668f4de9934f588a1802e11aac70b737c77
+size 2250054
diff --git a/resources/images/kmap_simple.jpg b/resources/images/kmap_simple.jpg
new file mode 100644
index 0000000..d31b045
--- /dev/null
+++ b/resources/images/kmap_simple.jpg
@@ -0,0 +1,3 @@
+version https://git-lfs.github.com/spec/v1
+oid sha256:78eab056f304fb021da9c265319e9279871eeae71dbe2e28b09bff0ed0ee4071
+size 83848