如何在Processing可视化中为条形图数据设置动画?

时间:2013-05-27 22:10:48

标签: loops processing data-visualization bar-chart

我正在试图弄清楚如何将这些行合并到draw()函数中的for循环中。基本上,我试图为条形图设置动画,例如尝试将每个条形从0绘制到它的Y值,然后绘制下一个条形图,依此类推。我可以手动添加下面的每个块,它可以很好地创建每一行,但似乎更好的自动化它。每行创建一个条/列。

float yPos = map(actions[1].presses, minVal, maxVal, height/2, 50);
line(lineXpos, actions[1].a, endLineXpos, actions[1].a);
actions[1].a = actions[1].a - 2;
if (actions[1].a < yPos) { 
  actions[1].a = height/2; 
}

float yPos2 = map(actions[2].presses, minVal, maxVal, height/2, 50);
line(lineXpos+10, actions[2].a, endLineXpos+10, actions[2].a);
actions[2].a = actions[2].a - 2;
if (actions[2].a < yPos2) { 
  actions[2].a = height/2; 
}

float yPos3 = map(actions[3].presses, minVal, maxVal, height/2, 50);
...
...
and so on...

似乎我可以将上面的内容合并到下面这样的内容中,但是当我运行它时,它会无休止地运行并创建一个斜坡。

void draw() {

for( int j = 1; j < actions.length; j++) {

actions[j].yPos = map(actions[j].presses, minVal, maxVal, height/2, 50);

line(actions[j].lineXpos, actions[j].a, actions[j].endLineXpos, actions[j].a);
actions[j].a = actions[j].a - 2; 

if (actions[j].a < actions[j].yPos) { 
  actions[j].a = height/2;
  actions[j].lineXpos = actions[j].lineXpos + 10;
  actions[j].endLineXpos = actions[j].endLineXpos + 10;
}

}

}

我认为行动[j] .yPos每次都会被覆盖。我有什么想法吗?

谢谢!

**编辑:在下面添加我的完整代码以进行测试。

Action[] actions;
int leng;
float minVal, maxVal;
float a;
float lineXpos = 0;
float endLineXpos = 10;

//now draw lines
float xPos = 0;
float lineX = 0;
float lineY = 0;

void setup() {

size(1400, 600);
background(160,196,242);
stroke(191,133,99);
fill(0);
smooth();

textSize(14);
textAlign(LEFT, TOP);

String[] data = loadStrings("keystrokes.log");

actions = new Action[data.length];

for(int i = 1; i < actions.length; i++) {
float[] values = float(split(data[i], ",")); 
actions[i] = new Action(values[0], values[1], values[2]);
if(i == 1) {
minVal = maxVal = actions[i].presses; 
}
if(actions[i].presses < minVal) minVal = actions[i].presses;
if(actions[i].presses > maxVal) maxVal = actions[i].presses;
}

println("clicks min:"+minVal+" max:"+maxVal);

}

void draw() {

for( int j = 1; j < actions.length; j++) {
//println("presses: "+actions[j].presses);
float yPos = map(actions[j].presses, minVal, maxVal, height/2, 50);

//draw lines
if(j == 1) {
  lineX = xPos;
  lineY = yPos;
} else {
   line(lineX+10, height/2, xPos, yPos); 
}
lineX = xPos;
lineY = yPos;
xPos += 10;
}

float yPos = map(actions[1].presses, minVal, maxVal, height/2, 50);
line(0, actions[1].a, 10, actions[1].a);
actions[1].a = actions[1].a - 2;
if (actions[1].a < yPos) { 
actions[1].a = height/2; 
}

float yPos2 = map(actions[2].presses, minVal, maxVal, height/2, 50);
line(0+10, actions[2].a, 10+10, actions[2].a);
actions[2].a = actions[2].a - 2;
if (actions[2].a < yPos2) { 
actions[2].a = height/2; 
}  

float yPos3 = map(actions[3].presses, minVal, maxVal, height/2, 50);
line(0+20, actions[3].a, 10+20, actions[3].a);
actions[3].a = actions[3].a - 2;
if (actions[3].a < yPos3) { 
actions[3].a = height/2; 
}

}

// the white dots Object, cookie cutter
class Action {
float time;
float clicks;
float presses;
float speed;
float a = height/2;
float xpos;
float lineXpos = 0;
float endLineXpos = 10;
float yPos;

Action(float t, float p, float c) {
time = t;
presses = p;
clicks = c;
speed = 2;
} 

}

Keystokes.log:

1369064940, 0, 0
1369065060, 65, 19
1369065180, 90, 28
1369065300, 179, 27
1369065420, 242, 20
1369065540, 156, 30

1 个答案:

答案 0 :(得分:1)

这将是for循环:

  float yPos = 0;
  for (int i = 1; i < actions.length; i++) {
    yPos = map(actions[i].presses, minVal, maxVal, height/2, 50);
    line((i-1)*10, actions[i].a, 10+(i-1)*10, actions[i].a);
    actions[i].a = actions[i].a - 2;
    if (actions[i].a < yPos) { 
      actions[i].a = height/2;
    }
  }

在考虑如何创建for循环时,有必要手动制作第一个,就像你已经完成的那样。然后,您可以使用创建每个条形的单个代码块之一,例如,这个:

  float yPos3 = map(actions[3].presses, minVal, maxVal, height/2, 50);
  line(0+20, actions[3].a, 10+20, actions[3].a);
  actions[3].a = actions[3].a - 2;
  if (actions[3].a < yPos3) { 
    actions[3].a = height/2;
  }

然后,您将为每个条形变化的变量变为通用变量。所以在这种情况下,你会发现你必须继续制作yPos1, yPos2, yPos3, etc.在你的for循环中,你只需要一个。在上面的for循环中,我在for循环之后立即声明yPos。然后,i将循环遍历actions[i]数组中的所有索引。最后,棘手的部分是确定绘制line(x1, y1, x2, y2)的位置的模式。我在原始代码中注意到,这些行是这样的:

line(0, actions[1].a, 10, actions[1].a);
line(0+10, actions[2].a, 10+10, actions[2].a);
line(0+20, actions[3].a, 10+20, actions[3].a);

所以我看到每次绘制一条线时都会增加10个像素。所以你可以把这一行写成:

line((i-1)*10, actions[i].a, 10+(i-1)*10, actions[i].a);

基本上,i从1到actions.length,即1, 2, 3, 4, 5, 6。并且您希望在0, 10, 20, 30, 40, 50上绘制您的线条。所以你可以写(i-1)*10。第二个x值的工作方式相同,但使用额外的10+

如果有帮助,请告诉我。