如何以编程方式生成具有不同颜色块的环/环

时间:2015-07-29 20:17:06

标签: image imagemagick dynamic-image-generation

一张图片胜过千言万语......

enter image description here

我想知道如何能够生成像这样的图像:(1)两个圆圈显然是完美的圆形,(2)我可以根据角度定义每个区域的开始和结束部分,例如第1节从垂直方向开始0弧度,从垂直方向开始以pi / 2弧度结束,(3)我可以定义每个区域的颜色。

事实上,戒指的外侧和内侧不应有黑色边框;每个区域的边界应与每个区域的颜色相同。

我怎么能用ImageMagick做到这一点?

3 个答案:

答案 0 :(得分:1)

您可以使用Arc命令在矢量图形中创建环。有关参数的详细信息,请参阅Mozilla的Path Document

使用ImageMagick,您可以-draw矢量图形的任何部分。例如:

convert -size 100x100 xc:transparent -stroke black -strokewidth 1 \
        -fill blue  -draw 'path "M 10 50 A 40 40 0 0 1 50 10 L 50 20 A 30 30 0 0 0 20 50 Z"' \
        -fill red   -draw 'path "M 50 10 A 40 40 0 0 1 90 50 L 80 50 A 30 30 0 0 0 50 20 Z"' \
        -fill green -draw 'path "M 90 50 A 40 40 0 0 1 10 50 L 20 50 A 30 30 0 0 0 80 50 Z"' \
        annulus.png

annulus

其他很棒的例子herehere

<强>更新

要创建更具编程性的方法,请使用任何OOP脚本语言。以下是Python&amp; amp;的快速示例Wand,但Ruby&amp;强烈推荐RMagick

#!/usr/bin/env python3

import math

from wand.color import Color
from wand.drawing import Drawing
from wand.image import Image

class Annulus(Image):
  def __init__(self, inner, outer, padding=5):
    self.Ri = inner
    self.Ro = outer
    side = (outer + padding) * 2
    self.midpoint = side/2
    super(Annulus, self).__init__(width=side,
                                  height=side,
                                  background=Color("transparent"))
    self.format = 'PNG'

  def __iadd__(self, segment):
    cos_start, cos_end = math.cos(segment.As), math.cos(segment.Ae)
    sin_start, sin_end = math.sin(segment.As), math.sin(segment.Ae)
    SiX, SiY = self.midpoint + self.Ri * cos_start, self.midpoint + self.Ri * sin_start
    SoX, SoY = self.midpoint + self.Ro * cos_start, self.midpoint + self.Ro * sin_start
    EiX, EiY = self.midpoint + self.Ri * cos_end,   self.midpoint +  self.Ri * sin_end
    EoX, EoY = self.midpoint + self.Ro * cos_end,   self.midpoint + self.Ro * sin_end
    with Drawing() as draw:
      for key, value in segment.draw_args.items():
        setattr(draw, key, value)
      draw.path_start()
      draw.path_move(to=(SiX, SiY))
      draw.path_elliptic_arc(to=(EiX, EiY),
                             radius=(self.Ri, self.Ri),
                             clockwise=True)
      draw.path_line(to=(EoX, EoY))
      draw.path_elliptic_arc(to=(SoX, SoY),
                             radius=(self.Ro, self.Ro),
                             clockwise=False)
      draw.path_close()
      draw.path_finish()
      draw(self)
    return self

class Segment(object):
  def __init__(self, start=0.0, end=0.0, **kwargs):
    self.As = start
    self.Ae = end
    self.draw_args = kwargs

if __name__ == '__main__':
  from wand.display import display
  ring  = Annulus(20, 40)
  ring += Segment(start=0,
                  end=math.pi/2,
                  fill_color=Color("yellow"))
  ring += Segment(start=math.pi/2,
                  end=math.pi,
                  fill_color=Color("pink"),
                  stroke_color=Color("magenta"),
                  stroke_width=1)
  ring += Segment(start=math.pi,
                  end=0,
                  fill_color=Color("lime"),
                  stroke_color=Color("orange"),
                  stroke_width=4)
  display(ring)

programmatic annulus

答案 1 :(得分:1)

我对gnuplot知之甚少,但认为它符合这里的法案 - 我的命令可能很粗糙,但它们看起来非常清晰有效。比我聪明的人可以改善它们!

无论如何,这是我提出的脚本:

set xrange [-1:1]
set yrange [-1:1]
set angles degrees
set size ratio -1
# r1 = annulus outer radius, r2 = annulus inner radius
r1=1.0
r2=0.8
unset border; unset tics; unset key; unset raxis
set terminal png size 1000,1000
set output 'output.png'
set style fill solid noborder
set object circle at first 0,0 front size r1 arc [0:60]    fillcolor rgb 'red'
set object circle at first 0,0 front size r1 arc [60:160]  fillcolor rgb 'green'
set object circle at first 0,0 front size r1 arc [160:360] fillcolor rgb 'blue'

# Splat a white circle on top to conceal central area
set object circle at first 0,0 front size r2 fillcolor rgb 'white'

plot -10 notitle

结果如下:

enter image description here

因此,如果您将上述脚本保存为annulus.cmd,您将运行它并使用命令

创建文件output.png
gnuplot annulus.cmd

显然,脚本的内容是3条线,它们以set object circle开始,每条线都以不同的颜色创建一个单独的环形段,具有不同的起点和终点角度。

涂鸦并改变一些事情会给出这个:

set xrange [-1:1]
set yrange [-1:1]
set angles degrees
set size ratio -1
# r1 = annulus outer radius, r2 = annulus inner radius
r1=1.0
r2=0.4
unset border; unset tics; unset key; unset raxis
set terminal png size 1000,1000
set output 'output.png'
set style fill solid noborder
set object circle at first 0,0 front size r1 arc [0:60]    fillcolor rgb 'red'
set object circle at first 0,0 front size r1 arc [60:120]  fillcolor rgb 'green'
set object circle at first 0,0 front size r1 arc [120:180] fillcolor rgb 'blue'
set object circle at first 0,0 front size r1 arc [180:240] fillcolor rgb 'yellow'
set object circle at first 0,0 front size r1 arc [240:300] fillcolor rgb 'black'
set object circle at first 0,0 front size r1 arc [300:360] fillcolor rgb 'magenta'

# Splat a white circle on top to conceal central area
set object circle at first 0,0 front size r2 fillcolor rgb 'white'

plot -10 notitle

enter image description here

答案 2 :(得分:1)

由于我更倾向于直线思考而不是圈子,我以为我会有另一种方式,一种完全不同的方式......

首先,将这个环形画出一条直线:

convert -size 45x40 xc:red xc:lime -size 270x40 xc:blue +append line.png

enter image description here

我偷偷摸摸地将线段的长度加起来为360,因此每度有一个像素 - 对于我简单的大脑来应对:-)所以,有45 px(度)的红色,45 px石灰(度)和270像素(度)的蓝色,它们都与+append一起附加在一起以形成线条。请注意,第一个-size 45x40设置会一直存在,直到稍后更改,因此在我将其更改为准备应用于蓝色之前,它会应用于红色和石灰线段。

现在我们围绕一个圆圈弯曲这条线,如下所示:

convert line.png -virtual-pixel White -distort arc 360 result.png

enter image description here

当你习惯了这个概念时,你也可以一气呵成:

convert -size 60x40 xc:red xc:lime xc:blue xc:cyan xc:magenta xc:black +append -virtual-pixel White -distort arc 360 result.png

enter image description here

您可以像这样在环形片段中添加灰色边框:

convert -size 600x400 xc:red xc:lime xc:blue xc:cyan xc:magenta xc:black -bordercolor "rgb(180,180,180)" -border 20 +append -virtual-pixel White -distort arc 360 result.png

enter image description here

如果您希望所有内容都在透明背景上,请将上面的white更改为none