AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / coding / 问题 / 78893279
Accepted
EJSABOLK
EJSABOLK
Asked: 2024-08-20 23:26:38 +0800 CST2024-08-20 23:26:38 +0800 CST 2024-08-20 23:26:38 +0800 CST

如何向 matplotlib contourf 颜色条添加额外的颜色条块

  • 772

我正在使用 matplotlib、xarray 和 cartopy 绘制一些数据的轮廓图。使用 ax.contourf 实例绘制数据轮廓图,生成下图中的颜色条。然后我屏蔽数组并填充所有等于零的值。

在此处输入图片描述

我想在颜色条的左侧添加一个额外的条,就像我在 GIMP 中构建的那样

在此处输入图片描述

我想象这会在 fig.colorbar() 调用之前执行,对原始 contourf 句柄进行操作,但无法弄清楚如何做到这一点。

谢谢任何建议。

最小样本如下,产生下图:

import matplotlib.pyplot as plt
import cartopy.crs as ccrs
import numpy as np
    
fig = plt.figure()
ax = plt.axes(projection = ccrs.Mollweide(central_longitude=96))
ax.coastlines()

X = np.arange(-180,180,1)
Y = np.arange(-90,91,1)
X_grid, Y_grid = np.meshgrid(X,Y)
Z = np.random.random_sample((len(Y),len(X))).round(1)
Z.sort()

levels = [0,0.05,0.25,0.5,0.75,0.85,1]
CS = ax.contourf(X_grid,Y_grid,Z,transform = ccrs.PlateCarree(), cmap='plasma', transform_first=True, levels=levels)
HS = ax.contourf(X_grid,Y_grid,np.ma.masked_not_equal(Z,0),transform=ccrs.PlateCarree(),colors='none', levels=[-0.01,0.01],hatches=['X','X'], transform_first=True)

cbar = fig.colorbar(CS, orientation='horizontal', aspect=40, location='top')

在此处输入图片描述 编辑:有一种解决方案我不喜欢,因为它不够稳健,即独立于颜色条定义阴影矩形,缩小颜色条,明确定义矩形补丁锚点和大小,然后将矩形添加到图中。我在以下代码片段中完成了此操作,生成了以下图像(请忽略我还对颜色条的比例或相对于颜色条的刻度位置进行了更改):

import matplotlib.patches as mpatches
hrect = mpatches.Rectangle((.0.9185, 0.8998), 0.057, 0.0395, ec='k',hatch='XXX', transform=fig.transFigure, figure=fig, fill=False)
fig.patches.extend([hrect])

在此处输入图片描述

这不是一个令人满意的答案,因为如果图形大小发生变化、地理轴发生变化等,矩形位置将相对于颜色条位置而变化。理想情况下,这个矩形将粘在颜色条对象的末尾,但这似乎是不可能的。在进行更改时,至少让所有内容保持一致的一种方法是使用 gridspec 定义特定的颜色条轴,然后在与 cax 相邻的单独 gridspec 子图中绘制矩形。同样,这并不优雅。

matplotlib
  • 1 1 个回答
  • 35 Views

1 个回答

  • Voted
  1. Best Answer
    JohanC
    2024-08-21T05:15:06+08:002024-08-21T05:15:06+08:00

    下面的方法添加了一个额外的级别来为阴影区域创建一个框。

    从最初使用的颜色中创建一个颜色列表,并扩展第一个区域的颜色。在示例代码中,使用白色为其着色,但您也可以使用与第一个区域相同的颜色。

    颜色列表用于创建ListedColormap(没有颜色图,则没有颜色图的信息)。

    创建等高线图和颜色条后,在第一个颜色条区域顶部绘制一个具有相同填充图案的透明矩形。

    import matplotlib.pyplot as plt
    from matplotlib.colors import ListedColormap
    import cartopy.crs as ccrs
    import numpy as np
    
    X = np.arange(-180, 180, 1)
    Y = np.arange(-90, 91, 1)
    X_grid, Y_grid = np.meshgrid(X, Y)
    Z = np.random.random_sample((len(Y), len(X))).round(1)
    Z.sort()
    
    fig = plt.figure()
    ax = plt.axes(projection=ccrs.Mollweide(central_longitude=96))
    ax.coastlines()
    
    hatch_level = 0.01
    levels = [0, hatch_level, 0.05, 0.25, 0.5, 0.75, 0.85, 1]
    colors = ['white'] + list(plt.get_cmap('plasma', len(levels) - 2).colors)
    cmap = ListedColormap(colors)
    CS = ax.contourf(X_grid, Y_grid, Z, transform=ccrs.PlateCarree(), cmap=cmap, transform_first=True, levels=levels)
    HS = ax.contourf(X_grid, Y_grid, np.ma.masked_not_equal(Z, 0), transform=ccrs.PlateCarree(), colors='none',
                     levels=[-hatch_level, hatch_level], hatches=['XX'], transform_first=True)
    HS.set_edgecolor('dodgerblue')  # will be used for hatch marks and the contour edges
    HS.set_linewidth(0)  # don't show the contour edges
    
    cbar = fig.colorbar(CS, orientation='horizontal', aspect=40, location='top')
    
    # create a rectangle on top of the first colored region
    rect = plt.Rectangle((0, 0), hatch_level, 1,  # x between 0 and hatch_level, y between 0 and 1
                         facecolor='none',  # make the facecolor transparent
                         hatch=HS.hatches[0],  # use the same hatch pattern as in HS
                         edgecolor=HS.get_edgecolor(),  # use the same hatch color as in HS
                         lw=0)  # don't show an edge around the rectanble
    # add the rectangle to the colorbar
    cbar.ax.add_patch(rect)
    
    # get the xtick labels of the colorbar
    xtick_labels = [t.get_text() for t in cbar.ax.get_xticklabels()]
    # replace the second label with the first, and empty the first
    xtick_labels[1] = xtick_labels[0]
    xtick_labels[0] = ''
    cbar.ax.set_xticklabels(xtick_labels)
    
    plt.show()
    

    具有额外阴影区域的轮廓图的颜色条

    如果阴影区域的表面颜色与第一个区域相同,则看起来会是这样的:

    # ...
    colors =  list(plt.get_cmap('plasma', len(levels) - 2).colors)
    colors = [colors[0] ]+ colors
    # ...
    HS.set_edgecolor('pink')
    

    阴影颜色与第一个区域相同

    • 1

相关问题

  • 二维散点图 - mcolors.Normalize 与 mcolors.LogNorm

  • 在 Streamlit 中渲染人口普查地图的正确方法

  • 使用 python 通过 matplotlib 在图表中添加自定义信息

  • 如何将 3D 直方图转换为热图

  • 为什么使用 freqz 并在 Octave 和 SciPy 之间进行比较时,绘图会翻转?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    Vue 3:创建时出错“预期标识符但发现‘导入’”[重复]

    • 1 个回答
  • Marko Smith

    为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行?

    • 1 个回答
  • Marko Smith

    具有指定基础类型但没有枚举器的“枚举类”的用途是什么?

    • 1 个回答
  • Marko Smith

    如何修复未手动导入的模块的 MODULE_NOT_FOUND 错误?

    • 6 个回答
  • Marko Smith

    `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它?

    • 3 个回答
  • Marko Smith

    何时应使用 std::inplace_vector 而不是 std::vector?

    • 3 个回答
  • Marko Smith

    在 C++ 中,一个不执行任何操作的空程序需要 204KB 的堆,但在 C 中则不需要

    • 1 个回答
  • Marko Smith

    PowerBI 目前与 BigQuery 不兼容:Simba 驱动程序与 Windows 更新有关

    • 2 个回答
  • Marko Smith

    AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String”

    • 1 个回答
  • Marko Smith

    我正在尝试仅使用海龟随机和数学模块来制作吃豆人游戏

    • 1 个回答
  • Martin Hope
    Aleksandr Dubinsky 为什么 InetAddress 上的 switch 模式匹配会失败,并出现“未涵盖所有可能的输入值”? 2024-12-23 06:56:21 +0800 CST
  • Martin Hope
    Phillip Borge 为什么这个简单而小的 Java 代码在所有 Graal JVM 上的运行速度都快 30 倍,但在任何 Oracle JVM 上却不行? 2024-12-12 20:46:46 +0800 CST
  • Martin Hope
    Oodini 具有指定基础类型但没有枚举器的“枚举类”的用途是什么? 2024-12-12 06:27:11 +0800 CST
  • Martin Hope
    sleeptightAnsiC `(表达式,左值) = 右值` 在 C 或 C++ 中是有效的赋值吗?为什么有些编译器会接受/拒绝它? 2024-11-09 07:18:53 +0800 CST
  • Martin Hope
    The Mad Gamer 何时应使用 std::inplace_vector 而不是 std::vector? 2024-10-29 23:01:00 +0800 CST
  • Martin Hope
    Chad Feller 在 5.2 版中,bash 条件语句中的 [[ .. ]] 中的分号现在是可选的吗? 2024-10-21 05:50:33 +0800 CST
  • Martin Hope
    Wrench 为什么双破折号 (--) 会导致此 MariaDB 子句评估为 true? 2024-05-05 13:37:20 +0800 CST
  • Martin Hope
    Waket Zheng 为什么 `dict(id=1, **{'id': 2})` 有时会引发 `KeyError: 'id'` 而不是 TypeError? 2024-05-04 14:19:19 +0800 CST
  • Martin Hope
    user924 AdMob:MobileAds.initialize() - 对于某些设备,“java.lang.Integer 无法转换为 java.lang.String” 2024-03-20 03:12:31 +0800 CST
  • Martin Hope
    MarkB 为什么 GCC 生成有条件执行 SIMD 实现的代码? 2024-02-17 06:17:14 +0800 CST

热门标签

python javascript c++ c# java typescript sql reactjs html

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve