{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import random\n",
    "import os,pickle\n",
    "\n",
    "\n",
    "def generate_triangle(radius, num_points):\n",
    "    \"\"\"生成一个空心三角形\"\"\"\n",
    "    angle = 2 * np.pi / 3\n",
    "    vertices = [(np.cos(i * angle) * radius, np.sin(i * angle) * radius) for i in range(3)]\n",
    "    \n",
    "    points = []\n",
    "    for i in range(3):\n",
    "        start = vertices[i]\n",
    "        end = vertices[(i + 1) % 3]\n",
    "        for j in range(num_points):\n",
    "            t = j / num_points\n",
    "            x = (1 - t) * start[0] + t * end[0]\n",
    "            y = (1 - t) * start[1] + t * end[1]\n",
    "            points.append((x, y))\n",
    "    \n",
    "    return points\n",
    "\n",
    "def generate_square(radius, num_points):\n",
    "    \"\"\"生成一个空心正方形\"\"\"\n",
    "    angle = np.pi / 2\n",
    "    vertices = [(np.cos(i * angle) * radius, np.sin(i * angle) * radius) for i in range(4)]\n",
    "    \n",
    "    points = []\n",
    "    for i in range(4):\n",
    "        start = vertices[i]\n",
    "        end = vertices[(i + 1) % 4]\n",
    "        for j in range(num_points):\n",
    "            t = j / num_points\n",
    "            x = (1 - t) * start[0] + t * end[0]\n",
    "            y = (1 - t) * start[1] + t * end[1]\n",
    "            points.append((x, y))\n",
    "    \n",
    "    return points\n",
    "\n",
    "def generate_circle(radius, num_points):\n",
    "    \"\"\"生成一个空心圆形\"\"\"\n",
    "    points = []\n",
    "    for i in range(num_points):\n",
    "        angle = 2 * np.pi * i / num_points\n",
    "        x = radius * np.cos(angle)\n",
    "        y = radius * np.sin(angle)\n",
    "        points.append((x, y))\n",
    "    \n",
    "    return points\n",
    "\n",
    "\n",
    "def generate_star(radius, num_points):\n",
    "    \"\"\"生成一个清晰的五角星轮廓\"\"\"\n",
    "    # 五角星的五个外部顶点（从圆上选出）\n",
    "    outer_vertices = [\n",
    "        (radius * np.cos(2 * np.pi * i / 5), radius * np.sin(2 * np.pi * i / 5))\n",
    "        for i in range(5)\n",
    "    ]\n",
    "    \n",
    "    # 按照五角星的顺序连接顶点\n",
    "    star_order = [0, 2, 4, 1, 3]  # 五角星的连接顺序\n",
    "    points = []\n",
    "\n",
    "    for i in range(5):\n",
    "        start = outer_vertices[star_order[i]]\n",
    "        end = outer_vertices[star_order[(i + 1) % 5]]\n",
    "        for j in range(num_points // 5):  # 每条边生成均匀的点\n",
    "            t = j / (num_points // 5)\n",
    "            x = (1 - t) * start[0] + t * end[0]\n",
    "            y = (1 - t) * start[1] + t * end[1]\n",
    "            points.append((x, y))\n",
    "    \n",
    "    return points\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def generate_heart(radius, num_points):\n",
    "    \"\"\"生成一个空心心形\"\"\"\n",
    "    points = []\n",
    "    for i in range(num_points):\n",
    "        t = 2 * np.pi * i / num_points  # 参数 t 在 [0, 2π] 之间均匀分布\n",
    "        x = radius * 16 * (np.sin(t) ** 3) / 15  # x 坐标\n",
    "        y = radius * (13 * np.cos(t) - 5 * np.cos(2 * t) - 2 * np.cos(3 * t) - np.cos(4 * t)) / 15 # y 坐标\n",
    "        points.append((x, y))\n",
    "    return points\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def add_concentrated_noise(points, noise_level=0.1, noise_ratio=0.1, num_clusters=5, img_num = 5):\n",
    "    noisy_points_list = []\n",
    "    for ii in range(img_num):\n",
    "        \"\"\"为图形添加集中噪声\"\"\"\n",
    "        noisy_points = points.copy()  # 保持原始点集\n",
    "        num_noise_points = int(len(points) * noise_ratio)  # 计算噪声点数\n",
    "        cluster_centers = []\n",
    "\n",
    "        # 随机选择噪声簇的中心\n",
    "        for _ in range(num_clusters):\n",
    "            cluster_x = random.uniform(min([x for x, y in points]), max([x for x, y in points]))\n",
    "            cluster_y = random.uniform(min([y for x, y in points]), max([y for x, y in points]))\n",
    "            cluster_centers.append((cluster_x, cluster_y))\n",
    "        \n",
    "        # 在每个簇中心附近生成噪声点\n",
    "        for _ in range(num_noise_points):\n",
    "            cluster_center = random.choice(cluster_centers)\n",
    "            noise_x = cluster_center[0] + random.uniform(-noise_level, noise_level)\n",
    "            noise_y = cluster_center[1] + random.uniform(-noise_level, noise_level)\n",
    "            noisy_points.append((noise_x, noise_y))\n",
    "        noisy_points_list.append(noisy_points)\n",
    "    return noisy_points_list\n",
    "\n",
    "\n",
    "\n",
    "# 创建保存图片的文件夹\n",
    "output_dir = '../imgs'\n",
    "if not os.path.exists(output_dir):\n",
    "    os.makedirs(output_dir)\n",
    "\n",
    "\n",
    "\n",
    "seed = 1234\n",
    "np.random.seed(seed)\n",
    "\n",
    "# 设置半径和点数\n",
    "\n",
    "\n",
    "def my_geom_dataset(radius=10,num_points=1000,my_noise_level=1,my_noise_ratio=0.1,my_num_cluster=2,img_num=10,noisy_geom_path=\"../my_dataset/Geom/\"):\n",
    "    # 生成图形\n",
    "    triangle_points = generate_triangle(radius, num_points)\n",
    "    square_points = generate_square(radius, num_points)\n",
    "    circle_points = generate_circle(radius, num_points)\n",
    "    star_points = generate_star(radius, num_points)\n",
    "    heart_points = generate_heart(radius,num_points)\n",
    "\n",
    "    points_list = [triangle_points] + [square_points] + [circle_points] + [star_points] + [heart_points]\n",
    "    weights_list = [np.ones(len(list(pp))) / len(list(pp)) for pp in points_list]\n",
    "    label_list = [i for i in range(5)]\n",
    "\n",
    "    # 添加噪声\n",
    "    triangle_noisy_list = add_concentrated_noise(triangle_points, noise_level=my_noise_level, noise_ratio=my_noise_ratio, num_clusters=my_num_cluster,img_num=img_num)\n",
    "    square_noisy_list = add_concentrated_noise(square_points, noise_level=my_noise_level, noise_ratio=my_noise_ratio, num_clusters=my_num_cluster,img_num=img_num)\n",
    "    circle_noisy_list = add_concentrated_noise(circle_points, noise_level=my_noise_level, noise_ratio=my_noise_ratio, num_clusters=my_num_cluster,img_num=img_num)\n",
    "    star_noisy_list = add_concentrated_noise(star_points, noise_level=my_noise_level, noise_ratio=my_noise_ratio, num_clusters=my_num_cluster,img_num=img_num)\n",
    "    heart_noisy_list = add_concentrated_noise(heart_points,noise_level=my_noise_level,noise_ratio=my_noise_ratio, num_clusters=my_num_cluster,img_num=img_num)\n",
    "\n",
    "    noisy_points_list = triangle_noisy_list + square_noisy_list + circle_noisy_list + star_noisy_list + heart_noisy_list\n",
    "    noisy_weights_list = [np.ones(len(list(pp))) / len(list(pp)) for pp in noisy_points_list]\n",
    "    noisy_label_list = [j for j in range(5) for i in range(img_num)]\n",
    "    \n",
    "    \n",
    "    # 保存数据到文件\n",
    "    os.makedirs(noisy_geom_path,exist_ok=True) \n",
    "    noisy_geom_file = noisy_geom_path + \"noisy_geom\" + str(img_num) + \"_\"+ str(num_points) + \"_\"+ str(my_num_cluster) + \"_\"+ str(my_noise_level)+ \"_\"+ str(my_noise_ratio)+ \"_\"+ str(seed) + \".pkl\"\n",
    "\n",
    "    with open(noisy_geom_file, 'wb') as f:\n",
    "        pickle.dump((points_list, weights_list, label_list,noisy_points_list,noisy_weights_list,noisy_label_list), f)\n",
    "    return points_list,weights_list,label_list,noisy_points_list,noisy_weights_list,noisy_label_list\n",
    "print(\"\")\n",
    "\n",
    "\n",
    "\n",
    "num_points = 100;my_noise_level=2;my_noise_ratio=0.1;my_num_cluster=5;img_num=50\n",
    "points_list, weights_list, label_list, noisy_points_list, noisy_weights_list, noisy_label_list = \\\n",
    "                my_geom_dataset(num_points=num_points,my_noise_level=my_noise_level,\n",
    "                                my_noise_ratio=my_noise_ratio,my_num_cluster=my_num_cluster,img_num=img_num)\n",
    "\n",
    "print()\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "def visualize_multiple_2d_lists_separated(points_lists, labels=None, title=\"2D Point Cloud Visualization\", separation=30):\n",
    "    \"\"\"\n",
    "    可视化多个二维点集，避免重叠\n",
    "    :param points_lists: list of 2D lists，每个元素是一个二维点集\n",
    "    :param labels: 每个点集的标签列表 (可选)\n",
    "    :param title: 可视化的标题\n",
    "    :param separation: 点集之间的间隔\n",
    "    \"\"\"\n",
    "    plt.figure(figsize=(12, 8))\n",
    "    \n",
    "    # 颜色列表，用于区分不同的点集\n",
    "    colors = ['r', 'g', 'b', 'c', 'm', 'y', 'k']\n",
    "    color_map = {i: colors[i % len(colors)] for i in range(len(points_lists))}\n",
    "    \n",
    "    for i, points in enumerate(points_lists):\n",
    "        points = np.array(points)  # 转换为 numpy 数组便于操作\n",
    "        \n",
    "        # 添加偏移量，使点集分开\n",
    "        offset = i * separation\n",
    "        points[:, 0] += offset  # 在 x 方向添加偏移\n",
    "        \n",
    "        label = labels[i] if labels else f\"Dataset {i + 1}\"\n",
    "        plt.scatter(points[:, 0], points[:, 1], s=10, label=label, c=color_map[i])\n",
    "    \n",
    "    plt.title(title)\n",
    "    plt.xlabel(\"X-axis\")\n",
    "    plt.ylabel(\"Y-axis\")\n",
    "    plt.axis('equal')  # 保持比例一致\n",
    "    # plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# 调用可视化函数\n",
    "visualize_multiple_2d_lists_separated(noisy_points_list[:5], title=\"Visualization of Multiple 2D Lists\")\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
