用ENIGMA-toolbox作图

89c088b2c3071ac5034c83178cff9caa.jpeg

之前一直使用ggseg呈现结果,最近想试一试其他绘图工具。ENIGMA-toolbox有所了解,绘图功能看起来门槛不高,所以就试着用它呈现一些结果。Matlab版本的ENIGMA-toolbox直接使用就是SurfStat的功能绘图,Python版本的绘图功能应该是根据SurfStat的绘图思路进行改写的。

  1. 下载和安装

参照官网不再赘述。需要注意的是该工具包依赖的numpy和pandas版本较老,若不想降级numpy和pandas的话,可以到工具包安装位置,使用VScode将工具包中以下内容进行修改。

  • numpy已经没有.float和.int的属性了,因此np.float 应该改成float, np.int改成int

  • pandas新版本不再使用error_bad_lines作为输入,因此需要将error_bad_lines=False改为on_bad_lines='skip'

当然另外一种办法就是conda建立虚拟环境,按要求安装依赖也是可行的。

2. 准备Desikan结果-->载入python-->根据标签匹配顺序-->绘图

绘制统计结果,只需要将其按照工具包所需的格式准备即可,具体而言就是脑区的顺序需要和ENIGMA-Toolbox保持一致。这个工具并不像ggseg一样是根据脑区的标签进行数据匹配的,因此使用该工具时需要确保标签和数据的顺序与工具包默认的顺序一致 x 3!

Desikan 68个区域(第一列结束后紧接第二列)

a220d2ef18be5dd312e5d1bbe2ef7032.png

皮层下区域(第一列结束后紧接第二列)

655add4b1c3f37e3b56ff736b3bbb26e.png

快速找到这些区域顺序的方式就是载入ENIGMA-Toolbox自带的一个结果文件

from enigmatoolbox.datasets import load_summary_stats
sum_stats = load_summary_stats('22q')
CT_vector = sum_stats['CortThick_case_vs_controls']['Structure']
SV_vector = sum_stats['SubVol_case_vs_controls']['Structure']

把它做成一个dataFrame用于匹配顺序

CT_order_df = pd.DataFrame(CT_vector.tolist(), columns=['Dependent Variable'])
SV_order_df = pd.DataFrame(SV_vector.tolist(), columns=['Dependent Variable'])

使用Merge的方式将载入的数据集,根据标签的顺序进行匹配。当然这不是必须的,如果将结果载入之前已经确保顺序没问题,就可以直接用数据作图。

CT_df_reordered = pd.merge(CT_order_df, CT_df, on='Dependent Variable', how='left')
SV_df_reordered = pd.merge(SV_order_df, SV_df, on='Dependent Variable', how='left')

获取需要呈现的统计值(这里是Cohen's d),然后使用parcel_to_surface函数将各个区域的数据映射到surface的vertex上面,就可以绘图了。

from enigmatoolbox.utils.parcellation import parcel_to_surface
from enigmatoolbox.plotting import plot_cortical, plot_subcortical
CT_d = CT_df_reordered['Cohen\'s d']
CT_d_fsa5 = parcel_to_surface(CT_d, 'aparc_fsa5')
# Project the results on the surface brain
plot_cortical(array_name=CT_d_fsa5,surface_name="fsa5",size=(1200, 600),cmap='RdBu_r',color_bar=False,color_range=(-1.8, 1.8),screenshot=True,filename=os.path.join(plot_path,'CT_d_all.png'),background=(1, 1, 1),transparent_bg=False,scale= (3,3))

4a55b3aa020d5737b63779183d0bb650.png

对于皮层下区域,由于区域比较少,不需要单独做parcel_to_surface,直接绘图即可。

SV_d = SV_df_reordered['Cohen\'s d']
plot_subcortical(array_name=SV_d, size=(1200, 600),cmap='RdBu_r', color_bar=False, color_range=(-1, 1),screenshot=True,filename=os.path.join(plot_path,'SV_d_all.png'),background=(1, 1, 1),transparent_bg=False,scale= (1,1))

869552fd62b9a191964862a7ba94165b.png

parcellated<->vertexwise的转换比想象中的简单,surface_to_parcel或者parcel_to_surface函数使用了一个vertex到ROI的映射表进行相互转换。

这是该工具包提供的映射表,有desikan, glasser360, schaefer100, 200,300,400, economo_koskinas,都是可以用该方式绘制和保存的,没有的模板也可以自己制作映射表绘制。

495870000b494acd918316d878c9e556.png

另外可以试用matlab的这个函数查看annot文件相关信息,判断映射表中区域的信息是否准确。

read_annotation('fsa5_lh_aparc.annot')

3. 保存

无论是plot_cortical还是plot_subcortical都默认使用了VTK生成图形,它们可以缩放和旋转,要保存图片需要将画图函数中的screenshot设置为True,并设置filename为保存文件的位置和名称。此外,由于后续有拼图的需要,建议将colorbar关闭,拼图完成后手动添加一个colorbar。

4. 拼图

Python拼图的工具好像不多(?!),此外plot_cortical和plot_subcortical函数都是基于VTK作图,保存是截图的方式,并不返回plt内容,大部分的拼图工具可能也不好使。所以建立一个简单工作流完成拼图:

  1. 读取图片

  2. subplot展示图片

  3. 裁剪(删除全白色的行和列)

    裁剪时如果想留白某一位置,可以将该位置加入非白元素。

  4. 保存

将上述步骤封装为函数方便多次调用。

def merge_images(images_dict, output_file, trim_edges=False):"""Merges a list of images into a single figure and saves it, based on provided image information.Parameters:- images_dict: List of dictionaries, each containing 'file_path' and 'title' for each image.- output_file: Path where the merged image should be saved.- trim_edges: Whether to trim white edges from the merged image before saving.# Example usage:images_dict = [{'file_path': os.path.join(plot_path, 'CT_d_all.png'), 'title': 'Cortical'},{'file_path': os.path.join(plot_path, 'SV_d_all.png'), 'title': 'Subcortical'}]output_file = os.path.join(plot_path, 'CT_SV_d_all.png')merge_images(images_dict, output_file, trim_edges=True)"""n_images = len(images_dict)fig, axs = plt.subplots(n_images, 1, figsize=(12, 7 * n_images))for i, img_info in enumerate(images_dict):img = mpimg.imread(img_info['file_path'])axs[i].imshow(img)axs[i].axis('off')title_text = img_info.get('title', '')axs[i].set_title(title_text, y=0.7, fontdict={'family': 'sans-serif', 'fontsize': 18},)# Adding a subtle marker (barely visible)axs[i].text(0.5, 0.65, '|', fontsize=12, color=(0.995, 0.995, 0.995), ha='center', transform=axs[i].transAxes)axs[i].text(0.5, 0.75, '|', fontsize=22, color=(0.995, 0.995, 0.995), ha='center', transform=axs[i].transAxes)plt.tight_layout()plt.savefig(output_file, bbox_inches='tight',dpi=300)if trim_edges:# Load the saved image, trim edges, and save againmerged_img = mpimg.imread(output_file)trimmed_img = trim_white_edges(merged_img)plt.imsave(output_file, np.ascontiguousarray(trimmed_img),dpi=300)def trim_white_edges(img_array):"""Trim white edges from an image with pixel values normalized to the 0 to 1 range.Parameters:- img_array: Image array with pixel values in the range [0, 1].Returns:- Cropped image array with white edges removed."""# Detect pixels that are not completely white (assuming white is [1, 1, 1])mask = np.any(img_array < 1, axis=-1)# Find the rows and columns that contain non-white pixelsrows = np.any(mask, axis=1)cols = np.any(mask, axis=0)# Crop the image to these rows and columnscropped_img = img_array[rows][:, cols]return cropped_img

调用函数即可拼图

images_dict = [{'file_path': os.path.join(plot_path, 'CT_d_all.png'), 'title': 'Cortical'},{'file_path': os.path.join(plot_path, 'SV_d_all.png'), 'title': 'Subcortical'}
]
output_file = os.path.join(plot_path, 'CT_SV_d_all.png')
merge_images(images_dict, output_file, trim_edges=True)

图中在subplot标题下加入了非白元素(肉眼难以识别),从而增加title和图片的距离;另外可以看到由于subplot标题的存在,左右半球之间的空白区域并未被移除,正好使得布局看起来更美观。美中不足的是皮层下区域看起来略大,所以在手动加colorbar的时候可以调一下大小。

43c930bdaf4782ff6d226682bd71dae1.png

该函数可以一次拼接多个png图片,比如以下病例-对照结果:

images_dict = [{'file_path': os.path.join(plot_path, 'CT_d_{}.png'.format(condition_name)), 'title': condition_name} for condition_name, _ in conditions]
output_file = os.path.join(plot_path, 'all_CT_d_maps.png')
merge_images(images_dict, output_file, trim_edges=True)

f7c78bf57f5c35b8d55e7a4f10a49a1f.png

往期推荐

ENIGMA-Toolbox

R|ggsegExtra绘制Desterieux统计结果

R| ggseg 绘制统计结果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/295901.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Qt 学习笔记】Qt 背景介绍

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ Qt 背景介绍 文章编号&#xff1a;Qt 学习笔记 / 01 文章目录 Qt 背景…

语音陪玩交友软件系统程序-app小程序H5三端源码交付,支持二开!

电竞行业的发展带动其周边产业的发展&#xff0c;绘制着游戏人物图画的抱枕、鼠标垫、海报销量极大&#xff0c;电竞游戏直播、游戏教程短视频也备受人们喜爱&#xff0c;自然&#xff0c;像游戏陪练、代练行业也随之生长起来&#xff0c;本文就来讲讲&#xff0c;从软件开发角…

Python读取Excel根据每行信息生成一个PDF——并自定义添加文本,可用于制作准考证

文章目录 有点小bug的:最终代码(无换行):有换行最终代码无bug根据Excel自动生成PDF,目录结构如上 有点小bug的: # coding=utf-8 import pandas as pd from reportlab.pdfgen import canvas from reportlab.lib.pagesizes import letter from reportlab.pdfbase import pdf…

STM32学习笔记(11_3)- 软件SPI读写W25Q64

无人问津也好&#xff0c;技不如人也罢&#xff0c;都应静下心来&#xff0c;去做该做的事。 最近在学STM32&#xff0c;所以也开贴记录一下主要内容&#xff0c;省的过目即忘。视频教程为江科大&#xff08;改名江协科技&#xff09;&#xff0c;网站jiangxiekeji.com 本期学…

蓝桥杯(更新中)

递归与递推 递归 1.指数型枚举 解析&#xff1a;从 1 ∼ n 这 n 个整数中随机选取任意多个&#xff0c;输出所有可能的选择方案。 思路&#xff1a;枚举每一位对应的数字选与不选&#xff0c;例如&#xff1a;第一位对应的数字为1&#xff0c;有一种方案是选1&#xff0c;另…

游戏引擎架构01__引擎架构图

根据游戏引擎架构预设的引擎架构来构建运行时引擎架构 ​

ES6学习(四)-- Reflect / Promise / Generator 函数 / Class

文章目录 1. Reflect1.1 代替Object 的某些方法1.2 修改某些Object 方法返回结果1.3 命令式变为函数行为1.4 ! 配合Proxy 2. ! Promise2.1 回调地狱2.2 Promise 使用2.3 Promise 对象的状态2.4 解决回调地狱的方法2.5 Promise.all2.6 Promise.race 3. Generator 函数3.1 基本语…

【现代企业管理】企业组织结构和组织文化的理论与实践——以华为为例

一、前言 管理是科学和艺术的统一体&#xff0c;它是企业成长的保证。企业管理中&#xff0c;管理者面对的往往不是一个完整的系统&#xff0c;而是各种不具有整体规律性的零碎信息的总和&#xff0c;因此进行信息的整合和研究是管理的重点和关键。 组织管理作为管理的四大职…

AGV无人驾驶跨境运输新模式引领未来物流

agv AGV即“自动导引运输车”&#xff0c;这一概念起源于欧美&#xff0c;在欧美及日韩市场的发展比较成熟&#xff0c;于上世纪末被引入国内。这种自动导引运输车可以广泛应用于汽车、化工、医药以及食品饮料等制造业场景&#xff0c;以及机场、码头等仓储物流行业场景&#x…

Redis入门到实战-第十九弹

Redis入门到实战 Redis中Count-min-sketch数据类型常见操作官网地址Redis概述Count-min-sketch常见操作更新计划 Redis中Count-min-sketch数据类型常见操作 完整命令参考官网 官网地址 声明: 由于操作系统, 版本更新等原因, 文章所列内容不一定100%复现, 还要以官方信息为准…

UE4 方块排序动画

【动画效果】 入动画&#xff1a; 出动画&#xff1a; 【分析】 入动画&#xff1a;方块动画排序方式为Z字形&#xff0c;堆砌方向为X和Y轴向 出动画&#xff1a;方块动画排序方式为随机 【关键蓝图】 1.构建方块砌体 2.入/出动画

【大模型】大模型 CPU 推理之 llama.cpp

【大模型】大模型 CPU 推理之 llama.cpp llama.cpp安装llama.cppMemory/Disk RequirementsQuantization测试推理下载模型测试 参考 llama.cpp 描述 The main goal of llama.cpp is to enable LLM inference with minimal setup and state-of-the-art performance on a wide var…

GPTs构建广告文案Agent(只需要一个网址链接即可生成文案及配图)

在大家已经有账号的前提下&#xff0c;我们来看看怎么做&#xff01;&#xff01;&#xff01; 进入GPTs的编辑界面 如下图&#xff1a; 如何配置呢&#xff1f; Name&#xff1a;给我们的GPTs起个名字。Description&#xff1a;简单介绍一下&#xff0c;我们创建的GPTs是…

亚马逊卧式婴儿车和坐式婴儿车需要那些认证?欧盟美国加拿大要求那些检测标准

卧式婴儿车和坐式婴儿车上亚马逊需要那些认证&#xff1f;欧盟美国加拿大要求那些检测标准&#xff1f; 很荣幸您能看到我的资讯&#xff0c;亚马逊各国认证是我公司的优势产品&#xff0c;确保申诉成功并正常销售。服务周到&#xff0c;速度快&#xff0c;周期短。 欧盟 卧…

消息存储与同步策略设计

消息存储与同步策略 https://github.com/robinfoxnan/BirdTalkServer 思路&#xff1a; 私聊写扩散&#xff0c;以用户为中心&#xff0c;存储2次&#xff1b;群聊读扩散&#xff0c;以群组为中心&#xff0c;存储一次&#xff1b;scylladb易于扩展&#xff0c;适合并发&…

双机 Cartogtapher 建图文件配置

双机cartogtapher建图 最近在做硕士毕设的最后一个实验&#xff0c;其中涉及到多机建图&#xff0c;经过调研最终采用cartographer建图算法&#xff0c;其中配置多机建图的文件有些麻烦&#xff0c;特此博客以记录 非常感谢我的同门 ”叶少“ 山上的稻草人-CSDN博客的帮助&am…

【2024红明谷】三道Web题目的记录

红明谷 文章目录 红明谷Web1 | SOLVED LaterWeb2 | UNSOLVEDWeb3 | SOLVED 容器已经关咯&#xff0c;所以有些场景只能靠回忆描述啦&#xff0c;学习为主&#xff0c;题目只是一个载体~ 本次比赛学习为主&#xff0c;确实再一次感受到久违的web题目的魅力了&#xff0c;可能也是…

3.java openCV4.x 入门-数据类型(CvType)与Scalar

专栏简介 &#x1f492;个人主页 &#x1f4f0;专栏目录 点击上方查看更多内容 &#x1f4d6;心灵鸡汤&#x1f4d6;我们唯一拥有的就是今天&#xff0c;唯一能把握的也是今天 &#x1f9ed;文章导航&#x1f9ed; ⬆️ 2.hello openCV ⬇️ 4.待更新 数据类型&#xff…

RD55UP06-V 三菱iQ-R系列C语言功能模块

RD55UP06-V 三菱iQ-R系列C语言功能模块 RD55UP06-V用户手册&#xff0c;RD55UP06-V功能&#xff0c;RD55UP06-V系统配置 RD55UP06-V参数规格&#xff1a;10BASE-T/100BASE-TX/1000BASE-T 1通道&#xff1b;字节存储次序格式小端模式; 可使用SD存储卡插槽&#xff1b;工作RAM 1…

结构体,联合体,枚举( 2 )

目录 2.联合体 2.1联合体类型的声明 2.2联合体的特点 2.3联合体的内存大小 3.枚举 3.1枚举类型的声明 3.2枚举类型的优点 3.3枚举类型的使用 2.联合体 联合体&#xff08;Union&#xff09;是另一种复合数据类型&#xff0c;它允许我们在同一内存位置存储不同的数据类型…