{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import trimesh\n",
    "import torch\n",
    "from torch_geometric.data import Data\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import trimesh\n",
    "import torch\n",
    "from torch_geometric.data import Data\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# --- Parameters ---\n",
    "# dir_path = './10k/train/2048/'    # folder containing OBJ files\n",
    "dir_path = './data/processed-car-pressure-data/data/'    # folder containing OBJ files\n",
    "dataset_name = 'car_frontForce_hdamp_test'\n",
    "\n",
    "num_files_to_read = -1  # number of files to read (can change)\n",
    "num_files_to_read = 10  # number of files to read (can change)\n",
    "\n",
    "meshes = []\n",
    "\n",
    "# --- Read OBJ files and create graphs ---\n",
    "files_geom = sorted([f for f in os.listdir(dir_path) if f.endswith('.ply')])\n",
    "files_press = sorted([f for f in os.listdir(dir_path) if f.endswith('.npy')])\n",
    "\n",
    "files_geom = files_geom[:num_files_to_read] if num_files_to_read > 0 else files_geom\n",
    "\n",
    "for geo_idx, filename in enumerate(files_geom[:]):\n",
    "    \n",
    "    print(filename, files_press[geo_idx])\n",
    "    \n",
    "    mesh = trimesh.load(os.path.join(dir_path, filename), process=False)\n",
    "\n",
    "    # Get vertices and faces\n",
    "    vertices = torch.tensor(mesh.vertices, dtype=torch.float)\n",
    "    faces = torch.tensor(mesh.faces, dtype=torch.long)\n",
    "    \n",
    "    press = np.load(dir_path + files_press[geo_idx])\n",
    "    press = press[-len(vertices):]\n",
    "\n",
    "    # Convert faces to edge indices\n",
    "    edge_index = torch.cat([\n",
    "        faces[:, [0, 1]], faces[:, [1, 2]], faces[:, [2, 0]]\n",
    "    ], dim=0).t().contiguous()  # shape: [2, num_edges]\n",
    "\n",
    "    # Remove duplicate edges\n",
    "    edge_index = torch.unique(edge_index, dim=1)\n",
    "\n",
    "    # Create PyTorch Geometric graph, now including 'face'\n",
    "    geo_idx = torch.tensor([geo_idx], dtype=torch.long)\n",
    "    data = Data(x = press, pos=vertices, edge_index=edge_index, face=faces.t().contiguous(), geo_idx=geo_idx)\n",
    "    meshes.append(data)\n",
    "    \n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# subdivide and solve\n",
    "\n",
    "import torch\n",
    "from torch_geometric.data import Data\n",
    "import trimesh\n",
    "import numpy as np\n",
    "from scipy.spatial import cKDTree\n",
    "\n",
    "\n",
    "def preprocess_graph(graph):\n",
    "    # Convert to numpy\n",
    "    vertices = graph.pos.cpu().numpy()\n",
    "    edge_index = graph.edge_index.cpu().numpy()\n",
    "\n",
    "    if hasattr(graph, 'face'):\n",
    "        faces = graph.face.cpu().numpy().T\n",
    "    else:\n",
    "        raise ValueError(\"Graph must contain a 'face' attribute with triangle indices\")\n",
    "    \n",
    "    # Build trimesh object\n",
    "    mesh = trimesh.Trimesh(vertices=vertices, faces=faces, process=False)\n",
    "\n",
    "    # Normalize mesh to fit bounding box within unit cube (longest dimension = 1)\n",
    "    bounds = mesh.bounds\n",
    "    size = bounds[1] - bounds[0]\n",
    "    max_dim = np.max(size)\n",
    "    if max_dim == 0:\n",
    "        raise ValueError(\"Mesh has zero-size bounding box\")\n",
    "    mesh.vertices = (mesh.vertices - bounds[0]) / max_dim\n",
    "\n",
    "    # Check for multiple disconnected components\n",
    "    if len(mesh.split(only_watertight=False)) > 1:\n",
    "        return None\n",
    "    if mesh.is_watertight == False:\n",
    "        print(\"### NOT WATERTIGHT###\")\n",
    "        return None\n",
    "    \n",
    "    # mesh = mesh.subdivide()\n",
    "    # mesh.vertices, mesh.faces = trimesh.remesh.subdivide(mesh.vertices, mesh.faces)\n",
    "\n",
    "    new_vertices = torch.from_numpy(mesh.vertices).float()\n",
    "    new_faces = torch.from_numpy(mesh.faces.T).long()\n",
    "\n",
    "    # Build edge_index from faces\n",
    "    # mesh.faces shape: (n_faces, 3)\n",
    "    edges = np.concatenate([\n",
    "        mesh.faces[:, [0, 1]],\n",
    "        mesh.faces[:, [1, 2]],\n",
    "        mesh.faces[:, [2, 0]]\n",
    "    ], axis=0)\n",
    "    \n",
    "    # Remove duplicate edges: sort each edge and find unique ones\n",
    "    edges_sorted = np.sort(edges, axis=1)\n",
    "    unique_edges, indices = np.unique(edges_sorted, axis=0, return_index=True)\n",
    "    \n",
    "    # Compute edge lengths\n",
    "    v = mesh.vertices\n",
    "    edge_lengths = np.linalg.norm(v[unique_edges[:, 0]] - v[unique_edges[:, 1]], axis=1)\n",
    "    \n",
    "    new_edge_index = torch.from_numpy(unique_edges.T).long()   # shape: [2, num_edges]\n",
    "    new_edge_attr = torch.from_numpy(edge_lengths).float()     # shape: [num_edges]\n",
    "\n",
    "    new_graph = Data(pos=new_vertices, face=new_faces,\n",
    "                     edge_index=new_edge_index, edge_attr=new_edge_attr,\n",
    "                     geo_idx=graph.geo_idx)\n",
    "\n",
    "    return new_graph\n",
    "\n",
    "graphs = []\n",
    "for i in range(len(meshes)):\n",
    "    print('processing graph', i)\n",
    "    graph_i = preprocess_graph(meshes[i])\n",
    "    if graph_i is not None:\n",
    "        graphs.append(graph_i)\n",
    " \n",
    "print('num_graphs', len(graphs))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "idx = 4\n",
    "print(meshes[idx])\n",
    "print(meshes[idx].x.shape)\n",
    "print(meshes[idx].pos.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "meshes\n",
    "\n",
    "import pyvista as pv\n",
    "\n",
    "# Enable notebook plotting (use 'static', 'client', 'panel', or 'pythreejs')\n",
    "pv.set_jupyter_backend('static')\n",
    "\n",
    "def plot_graph(graph, u, filename=None):\n",
    "    vertices = graph.pos.cpu().numpy()\n",
    "    faces = graph.face.cpu().numpy().T  # shape (n_faces, 3)\n",
    "\n",
    "    # PyVista expects faces as a flat array: [3, v0, v1, v2, 3, v0, v1, v2, ...]\n",
    "    faces_flat = np.hstack([np.full((faces.shape[0],1), 3), faces]).astype(np.int64).flatten()\n",
    "\n",
    "    # Create PolyData\n",
    "    mesh = pv.PolyData(vertices, faces_flat)\n",
    "    \n",
    "    print(len(u))\n",
    "    print(len(vertices))\n",
    "\n",
    "    mesh.point_data['helmholtz_solution'] = u[-len(vertices):]\n",
    "    # mesh.point_data['helmholtz_solution'] = u[:len(vertices)]\n",
    "    # Rotate the mesh if needed\n",
    "    # mesh.rotate_x(90)\n",
    "\n",
    "    # Create a plotter\n",
    "    p = pv.Plotter(notebook=True)\n",
    "\n",
    "    # Add mesh: choose colormap and colorbar location\n",
    "    p.add_mesh(\n",
    "        mesh, \n",
    "        scalars='helmholtz_solution', \n",
    "        # scalars='helmholtz_forcing', \n",
    "        cmap='coolwarm', \n",
    "        show_scalar_bar=True,\n",
    "        scalar_bar_args={\n",
    "            'title': 'Solution',\n",
    "            # 'title': 'Helmholtz Forcing',\n",
    "            'vertical': True,      # True = vertical colorbar; False = horizontal\n",
    "            'position_x': 0.1,    # X position in normalized [0,1]\n",
    "            'position_y': 0.1,     # Y position in normalized [0,1]\n",
    "            # 'position_x': 0.85,    # X position in normalized [0,1]\n",
    "            # 'position_y': 0.1,     # Y position in normalized [0,1]\n",
    "            'width': 0.05,\n",
    "            'height': 0.8\n",
    "        }\n",
    "    )\n",
    "    # p.show_axes()\n",
    "    p.show_grid()\n",
    "    p.camera_position = [np.array([2, -3, 2])*4, (0.25, 0.5, 0.25), (0, -1, 0)]\n",
    "    # p.show_axes()\n",
    "    p.show()\n",
    "    # p.save_graphic(dir_plt + 'graph_plot.pdf')\n",
    "idx = np.random.randint(0, len(meshes))\n",
    "plot_graph(meshes[idx], meshes[idx].x)\n",
    "plt.close()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# del meshes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ntimes = 12\n",
    "# graphs = [graph.clone() for graph in graphs for _ in range(ntimes)]\n",
    "\n",
    "# print(len(graphs))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "import scipy.sparse\n",
    "from scipy.sparse.linalg import LinearOperator, gmres, lgmres\n",
    "import matplotlib.pyplot as plt\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "\n",
    "# Example parameters: real wavenumber squared and damping\n",
    "# k_squared = 1000     # real part (frequency squared)\n",
    "k_squared = 500 # in draft\n",
    "# k_squared = 200     # real part (frequency squared)\n",
    "\n",
    "# c = -5.           \n",
    "# c = - 0.1 * k_squared #in draft\n",
    "c = 0.2 * k_squared #in draft\n",
    "\n",
    "\n",
    "\n",
    "def solve_helmholtz_equation(graph_data, VERBOSE=False):\n",
    "    V = graph_data['V']\n",
    "    F = graph_data['F']\n",
    "    rnd = graph_data['i']\n",
    "\n",
    "    # Build cotangent Laplacian\n",
    "    def cotangent_laplacian(vertices, faces):\n",
    "        n_vertices = vertices.shape[0]\n",
    "        I, J, W = [], [], []\n",
    "        for tri in faces:\n",
    "            pts = vertices[tri]\n",
    "            for i in range(3):\n",
    "                i1, i2, i3 = tri[i], tri[(i+1)%3], tri[(i+2)%3]\n",
    "                v1 = pts[(i+1)%3] - pts[i]\n",
    "                v2 = pts[(i+2)%3] - pts[i]\n",
    "                cot_angle = np.dot(v1, v2) / np.linalg.norm(np.cross(v1, v2))\n",
    "                I += [i2, i3]\n",
    "                J += [i3, i2]\n",
    "                W += [cot_angle/2, cot_angle/2]\n",
    "        L = scipy.sparse.coo_matrix((W, (I, J)), shape=(n_vertices, n_vertices))\n",
    "        L = L + L.T\n",
    "        diag = np.array(L.sum(axis=1)).flatten()\n",
    "        L = scipy.sparse.diags(diag) - L\n",
    "        return L\n",
    "\n",
    "    # Lumped mass matrix\n",
    "    def mass_matrix(vertices, faces):\n",
    "        n_vertices = vertices.shape[0]\n",
    "        M = np.zeros(n_vertices)\n",
    "        for tri in faces:\n",
    "            pts = vertices[tri]\n",
    "            area = np.linalg.norm(np.cross(pts[1]-pts[0], pts[2]-pts[0])) / 2\n",
    "            for idx in tri:\n",
    "                M[idx] += area / 3\n",
    "        return scipy.sparse.diags(M)\n",
    "\n",
    "    import time\n",
    "    start_time = time.time()\n",
    "    if VERBOSE: print('building cotangent Laplacian')\n",
    "    L = cotangent_laplacian(V, F).astype(np.complex128)\n",
    "    if VERBOSE: print('time:', time.time() - start_time, 's')\n",
    "\n",
    "    if VERBOSE: print('building lumped mass matrix')\n",
    "    M = mass_matrix(V, F).astype(np.complex128)\n",
    "    if VERBOSE: print('time:', time.time() - start_time, 's')\n",
    "\n",
    "    # Forcing function: localized Gaussian\n",
    "    np.random.seed(rnd)\n",
    "    \n",
    "    centre_cadidates = np.where(V[:, 2] < 1/5)[0]\n",
    "    center_f = np.random.choice(centre_cadidates)\n",
    "    \n",
    "    # center_f = np.random.randint(0, V.shape[0])\n",
    "    f = np.exp(-((V - V[center_f])**2).sum(axis=1) / (0.1**2)) * 0.1\n",
    "    f = f.astype(np.complex128)\n",
    "\n",
    "    # Build complex matrix: A = L + (k_squared + i * c) * M\n",
    "    complex_k_squared = - k_squared + 1j * c\n",
    "    A = L + complex_k_squared * M\n",
    "\n",
    "    if VERBOSE: print('assembling LinearOperator')\n",
    "    A_linop = LinearOperator(\n",
    "        matvec=lambda x: A @ x,\n",
    "        dtype=np.complex128,\n",
    "        shape=A.shape\n",
    "    )\n",
    "\n",
    "    if VERBOSE: print('solving...')\n",
    "    u, info = lgmres(A_linop, f, rtol=1e-5, maxiter=10000)\n",
    "\n",
    "    if VERBOSE: print(\"info:\", info)\n",
    "    if info != 0:\n",
    "        if VERBOSE: print('LGMRES did not converge')\n",
    "        u = None\n",
    "        \n",
    "    if VERBOSE: print('LGMRES Computing Time:', time.time() - start_time, 's')\n",
    "\n",
    "    return {\n",
    "        'u': u,\n",
    "        'f': f,\n",
    "        'center_f': center_f,\n",
    "        'k_squared': k_squared,\n",
    "        'c': c,\n",
    "        'total_time': time.time() - start_time\n",
    "    }\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "\n",
    "\n",
    "MULTIPROC = True\n",
    "# MULTIPROC = False\n",
    "\n",
    "\n",
    "if MULTIPROC == False:\n",
    "    print('### RUNNING SINGLE PROCESSING ###')\n",
    "    print('### NPROC:', 1)\n",
    "    \n",
    "    start_time = time.time()\n",
    "\n",
    "    graph_data = [ \n",
    "            {\n",
    "                'i':i,\n",
    "                'V': graph.pos.cpu().numpy(),\n",
    "                'F': graph.face.cpu().numpy().T\n",
    "            }\n",
    "            for i, graph in enumerate(graphs)\n",
    "        ]\n",
    "            \n",
    "\n",
    "    results = [solve_helmholtz_equation(graph) for graph in graph_data]\n",
    "\n",
    "    print('Done Computing Helmholtz Solutions')\n",
    "\n",
    "    graph_slns = []\n",
    "    times = []\n",
    "    for i, result in enumerate(results):\n",
    "        print('solved helmholtz', i)\n",
    "        if result['u'] is not None:\n",
    "            graph = graphs[i]\n",
    "            print('graph idx', graph.geo_idx.item())\n",
    "            graph.x = torch.from_numpy(result['u']).float()\n",
    "            graph.f = torch.from_numpy(result['f']).float()\n",
    "            graph.center_f = torch.tensor([result['center_f']]).int()\n",
    "            graph.c = torch.tensor([result['c']]).float()\n",
    "            graph_slns.append(graph)\n",
    "            times.append(result['total_time'])\n",
    "\n",
    "    print('num slns', len(graph_slns))\n",
    "    print('TOTAL RUNTIME:', time.time() - start_time, 's')\n",
    "\n",
    "\n",
    "if MULTIPROC == True:\n",
    "    import multiprocessing\n",
    "\n",
    "\n",
    "    def helmholtz_solver_mp(graphs, nproc=32):   \n",
    "        print('### RUNNING MULTIPROCESSING ###')\n",
    "        print('### NPROC:', nproc)\n",
    "        start_time = time.time()\n",
    "\n",
    "        # with multiprocessing.Pool(nproc) as pool:\n",
    "        #     # Parallel map over graphs\n",
    "        #     results = pool.map(worker, graphs)\n",
    "        \n",
    "        graph_data = [ \n",
    "            {\n",
    "                'i':i,\n",
    "                'V': graph.pos.cpu().numpy(),\n",
    "                'F': graph.face.cpu().numpy().T\n",
    "            }\n",
    "            for i, graph in enumerate(graphs)\n",
    "        ]\n",
    "\n",
    "        pool = multiprocessing.Pool(nproc)\n",
    "        results = pool.map(solve_helmholtz_equation, iter(graph_data))\n",
    "\n",
    "        print('Done Computing Helmholtz Solutions')\n",
    "\n",
    "        graph_slns = []\n",
    "        times = []\n",
    "        for i, result in enumerate(results):\n",
    "            print('solved helmholtz', i)\n",
    "            if result['u'] is not None:\n",
    "                graph = graphs[i]\n",
    "                print('graph idx', graph.geo_idx.item())\n",
    "                graph.x = torch.from_numpy(result['u']).float()\n",
    "                graph.f = torch.from_numpy(result['f']).float()\n",
    "                graph.center_f = torch.tensor([result['center_f']]).int()\n",
    "                graph.c = torch.tensor([result['c']]).float()\n",
    "                graph_slns.append(graph)\n",
    "                times.append(result['total_time'])\n",
    "\n",
    "        print('num slns', len(graph_slns))\n",
    "        print('TOTAL RUNTIME:', time.time() - start_time, 's')\n",
    "        \n",
    "        return graph_slns, times\n",
    "\n",
    "    graph_slns, times = helmholtz_solver_mp(graphs, nproc=32)\n",
    "\n",
    "del graphs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "dir_save = f'./data/helm_{dataset_name}/'\n",
    "import os\n",
    "import shutil\n",
    "if os.path.exists(dir_save):\n",
    "    shutil.rmtree(dir_save)\n",
    "os.makedirs(dir_save, exist_ok=False)\n",
    "        \n",
    "import pickle\n",
    "with open(dir_save + f'graphs.pkl', 'wb') as file:\n",
    "    pickle.dump(graph_slns, file)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "dir_save = f'./data/helm_{dataset_name}/'\n",
    "print(dir_save)\n",
    "with open(dir_save + f'graphs.pkl', 'rb') as file:\n",
    "    graph_slns = pickle.load(file)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(len(graph_slns))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pyvista as pv\n",
    "\n",
    "# Enable notebook plotting (use 'static', 'client', 'panel', or 'pythreejs')\n",
    "pv.set_jupyter_backend('static')\n",
    "\n",
    "def plot_graph(graph, u, filename=None):\n",
    "    vertices = graph.pos.cpu().numpy()\n",
    "    faces = graph.face.cpu().numpy().T  # shape (n_faces, 3)\n",
    "    u = u.cpu().numpy()\n",
    "\n",
    "    # PyVista expects faces as a flat array: [3, v0, v1, v2, 3, v0, v1, v2, ...]\n",
    "    faces_flat = np.hstack([np.full((faces.shape[0],1), 3), faces]).astype(np.int64).flatten()\n",
    "\n",
    "    # Create PolyData\n",
    "    mesh = pv.PolyData(vertices, faces_flat)\n",
    "    \n",
    "    mesh.point_data['helm'] = u\n",
    "    # mesh.point_data['helmholtz_forcing'] = graph.f.detach().cpu().numpy()\n",
    "    # Rotate the mesh if needed\n",
    "    # mesh.rotate_x(90)\n",
    "\n",
    "    # Create a plotter\n",
    "    p = pv.Plotter(notebook=True)\n",
    "\n",
    "    # Add mesh: choose colormap and colorbar location\n",
    "    # Find symmetric limits around zero\n",
    "    # abs_max = np.max(np.abs(u))\n",
    "\n",
    "    p.add_mesh(\n",
    "        mesh, \n",
    "        scalars='helm', \n",
    "        cmap='coolwarm', \n",
    "        # clim=[-abs_max, abs_max],  # Symmetric color limits\n",
    "        show_scalar_bar=True,\n",
    "        scalar_bar_args={\n",
    "            'title': 'helm',\n",
    "            'vertical': True,\n",
    "            'position_x': 0.1,\n",
    "            'position_y': 0.1,\n",
    "            'width': 0.05,\n",
    "            'height': 0.8\n",
    "        }\n",
    "    )\n",
    "    # p.show_axes()\n",
    "    p.show_grid()\n",
    "    p.camera_position = [np.array([2, 3, -2]), (0.0, 0.0, 0.0), (0, 0, 0)]\n",
    "    # p.show_axes()\n",
    "    p.show()\n",
    "    # p.save_graphic(dir_plt + 'graph_plot.pdf')\n",
    "idx = np.random.randint(0, len(graph_slns))\n",
    "plot_graph(graph_slns[idx], graph_slns[idx].x, filename='recon')\n",
    "plt.close()\n",
    "plot_graph(graph_slns[idx], graph_slns[idx].f, filename='recon')\n",
    "plt.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pyvista as pv\n",
    "import torch\n",
    "\n",
    "dir_save_vtp = f'./data/helm_{dataset_name}/vtp/'\n",
    "os.makedirs(dir_save_vtp, exist_ok=True)\n",
    "\n",
    "# Convert data to numpy\n",
    "def save_graph_vtp(graph, i):\n",
    "    \n",
    "    vertices = graph.pos.cpu().numpy()\n",
    "    faces = graph.face.cpu().numpy().T  # shape (n_faces, 3)\n",
    "\n",
    "    # PyVista expects faces as a flat array: [3, v0, v1, v2, 3, v0, v1, v2, ...]\n",
    "    faces_flat = np.hstack([np.full((faces.shape[0],1), 3), faces]).astype(np.int64).flatten()\n",
    "\n",
    "    # Create PolyData\n",
    "    mesh = pv.PolyData(vertices, faces_flat)\n",
    "\n",
    "    # Add solution as point data\n",
    "    mesh.point_data['helmholtz_solution'] = graph.x.cpu().numpy()\n",
    "    mesh.point_data['helmholtz_forcing'] =  graph.f.cpu().numpy()\n",
    "    mesh.point_data['helmholtz_c'] =  graph.x.cpu().numpy() ** 0. * graph.c.cpu().numpy()\n",
    "    num = graph.geo_idx.item()  # Get the geo_idx as an integer\n",
    "    # Save to VTK\n",
    "    mesh.save(dir_save_vtp + f\"helmholtz_solution_{i}_{num}.vtp\")\n",
    "    print(dir_save_vtp + f\"Saved to helmholtz_solution_{i}_{num}.vtp\")\n",
    "    \n",
    "idx_save = range(len(graph_slns))\n",
    "for i in idx_save:\n",
    "    save_graph_vtp(graph_slns[i], i)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "gabi",
   "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.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
