// ConvexSquare.cpp : アプリケーションのエントリ ポイントを定義します。
//
#define WIN32_LEAN_AND_MEAN // Windows ヘッダーからほとんど使用されていない部分を除外する
// Windows ヘッダー ファイル
#include <windows.h>
// C ランタイム ヘッダー ファイル
#include <tchar.h>
#include <iterator>
#include <algorithm>
#include <vector>
std::vector<int> v{ 136, 577, 110, 927, 472, 199, 157, 808, 388, 598, 94, 31, 388, 157, 325, 409, 787, 897, 850, 598 };
struct vec2d {
int x, y;
vec2d& operator=(const vec2d& ref) {
x = ref.x;
y = ref.y;
return *this;
}
bool operator==(const vec2d& ref) const {
return x == ref.x && y == ref.y;
}
bool operator!=(const vec2d& ref) const {
return !operator==(ref);
}
vec2d operator-(const vec2d& ref) const {
return { x - ref.x, y - ref.y };
}
vec2d operator+(const vec2d& ref) const {
return { x + ref.x, y + ref.y };
}
int dot(const vec2d& ref) const {
return x * ref.y + y * ref.x;
}
int cross(const vec2d& ref) const {
return x * ref.y - ref.x * y;
}
};
using Vertices = std::vector<vec2d>;
auto getVertices()
{
auto i = std::begin(v);
auto e = i; std::advance(e, std::size(v) - std::size(v) % 2);
Vertices result;
while (i != e) {
result.push_back({ *i, *std::next(i) });
std::advance(i, 2);
}
return result;
}
int findStartPoint(const Vertices& vertices)
{
auto p = 0;
auto x = vertices[p].x, y = vertices[p].y;
for (size_t i = 0; i < std::size(vertices); i++) {
auto diff = y - vertices[i].y;
if (diff > 0) {
y = vertices[i].y;
x = vertices[i].x;
p = i;
}
else if (diff == 0 && x > vertices[i].x) {
x = vertices[i].x;
p = i;
}
}
return p;
}
auto getConvexHull(const Vertices& vertices)
{
Vertices result;
if (std::size(vertices) == 0) {
return result;
}
auto a = findStartPoint(vertices);
for (;;) {
result.push_back(vertices[a]);
auto b = 0;
for (size_t i = 1; i < std::size(vertices); i++) {
auto c = i;
if (vertices[b] == vertices[a]) {
b = c;
}
else {
vec2d A = vertices[a];
vec2d B = vertices[b];
vec2d C = vertices[c];
int v = (B - A).cross(C - A);
if (v > 0) {
b = c;
}
}
}
a = b;
if (result[0] == vertices[a]) {
break;
}
}
return result;
}
auto getInclusionCircle(const Vertices& vertices)
{
Vertices result;
return result;
}
// グローバル変数:
HINSTANCE hInst; // 現在のインターフェイス
WCHAR szTitle[]=_T("ConvexSquare"); // タイトル バーのテキスト
WCHAR szWindowClass[]=_T("CONVEXSQUARE"); // メイン ウィンドウ クラス名
// このコード モジュールに含まれる関数の宣言を転送します:
ATOM MyRegisterClass(HINSTANCE hInstance);
BOOL InitInstance(HINSTANCE, int);
LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK About(HWND, UINT, WPARAM, LPARAM);
int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
_In_opt_ HINSTANCE hPrevInstance,
_In_ LPWSTR lpCmdLine,
_In_ int nCmdShow)
{
UNREFERENCED_PARAMETER(hPrevInstance);
UNREFERENCED_PARAMETER(lpCmdLine);
// TODO: ここにコードを挿入してください。
auto vertices = getVertices();
auto convex_hull = getConvexHull(vertices);
// グローバル文字列を初期化する
MyRegisterClass(hInstance);
// アプリケーション初期化の実行:
if (!InitInstance (hInstance, nCmdShow))
{
return FALSE;
}
MSG msg;
// メイン メッセージ ループ:
while (GetMessage(&msg, nullptr, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return (int) msg.wParam;
}
//
// 関数: MyRegisterClass()
//
// 目的: ウィンドウ クラスを登録します。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
WNDCLASSEXW wcex;
wcex.cbSize = sizeof(WNDCLASSEX);
wcex.style = CS_HREDRAW | CS_VREDRAW;
wcex.lpfnWndProc = WndProc;
wcex.cbClsExtra = 0;
wcex.cbWndExtra = 0;
wcex.hInstance = hInstance;
wcex.hIcon = LoadIcon(nullptr, IDI_APPLICATION);
wcex.hCursor = LoadCursor(nullptr, IDC_ARROW);
wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW+1);
wcex.lpszMenuName = NULL;
wcex.lpszClassName = szWindowClass;
wcex.hIconSm = LoadIcon(nullptr, IDI_APPLICATION);
return RegisterClassExW(&wcex);
}
//
// 関数: InitInstance(HINSTANCE, int)
//
// 目的: インスタンス ハンドルを保存して、メイン ウィンドウを作成します
//
// コメント:
//
// この関数で、グローバル変数でインスタンス ハンドルを保存し、
// メイン プログラム ウィンドウを作成および表示します。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
hInst = hInstance; // グローバル変数にインスタンス ハンドルを格納する
HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr);
if (!hWnd)
{
return FALSE;
}
ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);
return TRUE;
}
//
// 関数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
// 目的: メイン ウィンドウのメッセージを処理します。
//
// WM_COMMAND - アプリケーション メニューの処理
// WM_PAINT - メイン ウィンドウを描画する
// WM_DESTROY - 中止メッセージを表示して戻る
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
switch (message)
{
case WM_LBUTTONDOWN:
{
int x = LOWORD(lParam);
int y = HIWORD(lParam);
v.push_back({ x });
v.push_back({ y });
::InvalidateRect(hWnd, nullptr, true);
}
break;
case WM_RBUTTONDOWN:
{
v.clear();
::InvalidateRect(hWnd, nullptr, true);
}
break;
case WM_MBUTTONDOWN:
{
std::vector<int> tmp{ 136, 577, 110, 927, 472, 199, 157, 808, 388, 598, 94, 31, 388, 157, 325, 409, 787, 897, 850, 598 };
v.swap(tmp);
::InvalidateRect(hWnd, nullptr, true);
}
break;
case WM_PAINT:
{
PAINTSTRUCT ps;
HDC hdc = BeginPaint(hWnd, &ps);
// TODO: HDC を使用する描画コードをここに追加してください...
::Rectangle(hdc, 0, 0, 999, 999);
{
Vertices vertices = getVertices();
for (auto& v : vertices) {
::Rectangle(hdc, v.x - 1, v.y - 1, v.x + 1, v.y + 1);
}
}
{
Vertices vertices = getConvexHull(getVertices());
if (std::size(vertices) > 0) {
{
for (size_t i = 0; i < std::size(vertices); i++) {
char txt[255];
sprintf_s(txt, 255, "%zu", i);
::TextOutA(hdc, vertices[i].x, vertices[i].y, txt, strlen(txt));
}
}
{
HBRUSH red = ::CreateSolidBrush(RGB(255, 0, 0));
HGDIOBJ brush = ::SelectObject(hdc, red);
HGDIOBJ pen = ::SelectObject(hdc, ::GetStockObject(NULL_PEN));
for (auto& v : vertices) {
::Rectangle(hdc, v.x - 2, v.y - 2, v.x + 2, v.y + 2);
}
::SelectObject(hdc, pen);
::SelectObject(hdc, brush);
::DeleteObject(red);
}
{
auto i = std::begin(vertices);
auto e = std::end(vertices);
::MoveToEx(hdc, i->x, i->y, nullptr);
while (i != e) {
auto to = std::next(i);
if (to == e)
to = std::begin(vertices);
::LineTo(hdc, to->x, to->y);
i++;
}
}
}
}
EndPaint(hWnd, &ps);
}
break;
case WM_DESTROY:
PostQuitMessage(0);
break;
default:
return DefWindowProc(hWnd, message, wParam, lParam);
}
return 0;
}
// ConvexSquare.cpp : アプリケーションのエントリ ポイントを定義します。
//

#define WIN32_LEAN_AND_MEAN             // Windows ヘッダーからほとんど使用されていない部分を除外する
// Windows ヘッダー ファイル
#include <windows.h>
// C ランタイム ヘッダー ファイル
#include <tchar.h>

#include <iterator>
#include <algorithm>
#include <vector>

std::vector<int> v{ 136, 577, 110, 927, 472, 199, 157, 808, 388, 598, 94, 31, 388, 157, 325, 409, 787, 897, 850, 598 };

struct vec2d {
	int x, y;

	vec2d& operator=(const vec2d& ref) {
		x = ref.x;
		y = ref.y;
		return *this;
	}

	bool operator==(const vec2d& ref) const {
		return x == ref.x && y == ref.y;
	}

	bool operator!=(const vec2d& ref) const {
		return !operator==(ref);
	}

	vec2d operator-(const vec2d& ref) const {
		return { x - ref.x, y - ref.y };
	}
	vec2d operator+(const vec2d& ref) const {
		return { x + ref.x, y + ref.y };
	}

	int dot(const vec2d& ref) const {
		return x * ref.y + y * ref.x;
	}

	int cross(const vec2d& ref) const {
		return x * ref.y - ref.x * y;
	}
};

using Vertices = std::vector<vec2d>;

auto getVertices()
{
	auto i = std::begin(v);
	auto e = i; std::advance(e, std::size(v) - std::size(v) % 2);
	Vertices result;
	while (i != e) {
		result.push_back({ *i, *std::next(i) });
		std::advance(i, 2);
	}
	return result;
}

int findStartPoint(const Vertices& vertices)
{
	auto p = 0;
	auto x = vertices[p].x, y = vertices[p].y;
	for (size_t i = 0; i < std::size(vertices); i++) {
		auto diff = y - vertices[i].y;
		if (diff > 0) {
			y = vertices[i].y;
			x = vertices[i].x;
			p = i;
		}
		else if (diff == 0 && x > vertices[i].x) {
			x = vertices[i].x;
			p = i;
		}
	}
	return p;
}

auto getConvexHull(const Vertices& vertices)
{
	Vertices result;
	if (std::size(vertices) == 0) {
		return result;
	}
	auto a = findStartPoint(vertices);
	for (;;) {
		result.push_back(vertices[a]);
		auto b = 0;
		for (size_t i = 1; i < std::size(vertices); i++) {
			auto c = i;
			if (vertices[b] == vertices[a]) {
				b = c;
			}
			else {
				vec2d A = vertices[a];
				vec2d B = vertices[b];
				vec2d C = vertices[c];
				int v = (B - A).cross(C - A);
				if (v > 0) {
					b = c;
				}
			}
		}
		a = b;
		if (result[0] == vertices[a]) {
			break;
		}
	}

	return result;
}

auto getInclusionCircle(const Vertices& vertices)
{
	Vertices result;

	return result;
}

// グローバル変数:
HINSTANCE hInst;                                // 現在のインターフェイス
WCHAR szTitle[]=_T("ConvexSquare");                  // タイトル バーのテキスト
WCHAR szWindowClass[]=_T("CONVEXSQUARE");            // メイン ウィンドウ クラス名

// このコード モジュールに含まれる関数の宣言を転送します:
ATOM                MyRegisterClass(HINSTANCE hInstance);
BOOL                InitInstance(HINSTANCE, int);
LRESULT CALLBACK    WndProc(HWND, UINT, WPARAM, LPARAM);
INT_PTR CALLBACK    About(HWND, UINT, WPARAM, LPARAM);

int APIENTRY wWinMain(_In_ HINSTANCE hInstance,
                     _In_opt_ HINSTANCE hPrevInstance,
                     _In_ LPWSTR    lpCmdLine,
                     _In_ int       nCmdShow)
{
    UNREFERENCED_PARAMETER(hPrevInstance);
    UNREFERENCED_PARAMETER(lpCmdLine);

    // TODO: ここにコードを挿入してください。
	auto vertices = getVertices();
	auto convex_hull = getConvexHull(vertices);

    // グローバル文字列を初期化する
    MyRegisterClass(hInstance);

    // アプリケーション初期化の実行:
    if (!InitInstance (hInstance, nCmdShow))
    {
        return FALSE;
    }

    MSG msg;

    // メイン メッセージ ループ:
    while (GetMessage(&msg, nullptr, 0, 0))
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    return (int) msg.wParam;
}



//
//  関数: MyRegisterClass()
//
//  目的: ウィンドウ クラスを登録します。
//
ATOM MyRegisterClass(HINSTANCE hInstance)
{
    WNDCLASSEXW wcex;

    wcex.cbSize = sizeof(WNDCLASSEX);

    wcex.style          = CS_HREDRAW | CS_VREDRAW;
    wcex.lpfnWndProc    = WndProc;
    wcex.cbClsExtra     = 0;
    wcex.cbWndExtra     = 0;
    wcex.hInstance      = hInstance;
    wcex.hIcon          = LoadIcon(nullptr, IDI_APPLICATION);
    wcex.hCursor        = LoadCursor(nullptr, IDC_ARROW);
    wcex.hbrBackground  = (HBRUSH)(COLOR_WINDOW+1);
    wcex.lpszMenuName   = NULL;
    wcex.lpszClassName  = szWindowClass;
    wcex.hIconSm        = LoadIcon(nullptr, IDI_APPLICATION);

    return RegisterClassExW(&wcex);
}

//
//   関数: InitInstance(HINSTANCE, int)
//
//   目的: インスタンス ハンドルを保存して、メイン ウィンドウを作成します
//
//   コメント:
//
//        この関数で、グローバル変数でインスタンス ハンドルを保存し、
//        メイン プログラム ウィンドウを作成および表示します。
//
BOOL InitInstance(HINSTANCE hInstance, int nCmdShow)
{
   hInst = hInstance; // グローバル変数にインスタンス ハンドルを格納する

   HWND hWnd = CreateWindowW(szWindowClass, szTitle, WS_OVERLAPPEDWINDOW,
      CW_USEDEFAULT, 0, CW_USEDEFAULT, 0, nullptr, nullptr, hInstance, nullptr);

   if (!hWnd)
   {
      return FALSE;
   }

   ShowWindow(hWnd, nCmdShow);
   UpdateWindow(hWnd);

   return TRUE;
}

//
//  関数: WndProc(HWND, UINT, WPARAM, LPARAM)
//
//  目的: メイン ウィンドウのメッセージを処理します。
//
//  WM_COMMAND  - アプリケーション メニューの処理
//  WM_PAINT    - メイン ウィンドウを描画する
//  WM_DESTROY  - 中止メッセージを表示して戻る
//
//
LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    switch (message)
    {
	case WM_LBUTTONDOWN:
		{
			int x = LOWORD(lParam);
			int y = HIWORD(lParam);
			v.push_back({ x });
			v.push_back({ y });
			::InvalidateRect(hWnd, nullptr, true);
		}
		break;
	case WM_RBUTTONDOWN:
		{
			v.clear();
			::InvalidateRect(hWnd, nullptr, true);
		}
		break;
	case WM_MBUTTONDOWN:
		{
			std::vector<int> tmp{ 136, 577, 110, 927, 472, 199, 157, 808, 388, 598, 94, 31, 388, 157, 325, 409, 787, 897, 850, 598 };
			v.swap(tmp);
			::InvalidateRect(hWnd, nullptr, true);
		}
		break;
	case WM_PAINT:
        {
            PAINTSTRUCT ps;
            HDC hdc = BeginPaint(hWnd, &ps);
            // TODO: HDC を使用する描画コードをここに追加してください...

			::Rectangle(hdc, 0, 0, 999, 999);

			{
				Vertices vertices = getVertices();
				for (auto& v : vertices) {
					::Rectangle(hdc, v.x - 1, v.y - 1, v.x + 1, v.y + 1);
				}
			}

			{
				Vertices vertices = getConvexHull(getVertices());
				if (std::size(vertices) > 0) {
					{
						for (size_t i = 0; i < std::size(vertices); i++) {
							char txt[255];
							sprintf_s(txt, 255, "%zu", i);
							::TextOutA(hdc, vertices[i].x, vertices[i].y, txt, strlen(txt));
						}
					}

					{
						HBRUSH red = ::CreateSolidBrush(RGB(255, 0, 0));
						HGDIOBJ brush = ::SelectObject(hdc, red);
						HGDIOBJ pen = ::SelectObject(hdc, ::GetStockObject(NULL_PEN));
						for (auto& v : vertices) {
							::Rectangle(hdc, v.x - 2, v.y - 2, v.x + 2, v.y + 2);
						}
						::SelectObject(hdc, pen);
						::SelectObject(hdc, brush);
						::DeleteObject(red);
					}
					{
						auto i = std::begin(vertices);
						auto e = std::end(vertices);
						::MoveToEx(hdc, i->x, i->y, nullptr);
						while (i != e) {
							auto to = std::next(i);
							if (to == e)
								to = std::begin(vertices);
							::LineTo(hdc, to->x, to->y);
							i++;
						}
					}
				}
			}

            EndPaint(hWnd, &ps);
        }
        break;
    case WM_DESTROY:
        PostQuitMessage(0);
        break;
    default:
        return DefWindowProc(hWnd, message, wParam, lParam);
    }
    return 0;
}

