Skip to content

Commit ea4b3b5

Browse files
Merge pull request #373 from plotly/fix_issues_and_awesome_improve_the_bar3_functionality
Fix issues and awesome improve the bar3 functionality
2 parents aefe991 + bee862f commit ea4b3b5

File tree

2 files changed

+366
-1
lines changed

2 files changed

+366
-1
lines changed

plotly/plotlyfig_aux/core/updateData.m

+3-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,9 @@
2222
elseif strcmpi(obj.PlotOptions.TreatAs, 'polarhistogram')
2323
updateHistogramPolar(obj, dataIndex);
2424
elseif strcmpi(obj.PlotOptions.TreatAs, 'coneplot')
25-
updateConeplot(obj, dataIndex);
25+
updateConeplot(obj, dataIndex);
26+
elseif strcmpi(obj.PlotOptions.TreatAs, 'bar3')
27+
updateBar3(obj, dataIndex);
2628

2729
% this one will be revomed
2830
elseif strcmpi(obj.PlotOptions.TreatAs, 'streamtube')
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,363 @@
1+
function obj = updateBar3(obj, surfaceIndex)
2+
3+
%-AXIS INDEX-%
4+
axIndex = obj.getAxisIndex(obj.State.Plot(surfaceIndex).AssociatedAxis);
5+
6+
%-CHECK FOR MULTIPLE AXES-%
7+
[xsource, ysource] = findSourceAxis(obj,axIndex);
8+
9+
%-SURFACE DATA STRUCTURE- %
10+
bar_data = get(obj.State.Plot(surfaceIndex).Handle);
11+
figure_data = get(obj.State.Figure.Handle);
12+
13+
%-AXIS STRUCTURE-%
14+
axis_data = get(ancestor(bar_data.Parent,'axes'));
15+
16+
%-GET SCENE-%
17+
eval(['scene = obj.layout.scene' num2str(xsource) ';']);
18+
19+
%-------------------------------------------------------------------------%
20+
21+
%-associate scene-%
22+
obj.data{surfaceIndex}.scene = sprintf('scene%d', xsource);
23+
24+
%-------------------------------------------------------------------------%
25+
26+
%-surface type-%
27+
obj.data{surfaceIndex}.type = 'mesh3d';
28+
29+
%-------------------------------------------------------------------------%
30+
31+
%-FORMAT DATA-%
32+
xdata = bar_data.XData;
33+
ydata = bar_data.YData;
34+
zdata = bar_data.ZData;
35+
cdata = bar_data.CData;
36+
37+
%-parse xedges-%
38+
xedges = xdata(2, 1:2:end);
39+
40+
%-parse yedges-%
41+
yedges = ydata(2:6:end, 2);
42+
yedges = [yedges', mean(diff(yedges(1:2)))];
43+
44+
%-parse values-%
45+
values = [];
46+
for n = 1:6:size(zdata, 1)
47+
values = [values, diff(zdata(n:n+1, 2))];
48+
end
49+
50+
%-parse offsets-%
51+
offsets = zdata(1:6:end, 2)';
52+
53+
%-------------------------------------------------------------------------%
54+
55+
%-get the values to use plotly's mesh3D-%
56+
bargap = diff(yedges(1:2)) - diff(ydata(2:3));
57+
[X, Y, Z, I, J, K] = get_plotly_mesh3d(xedges, yedges, values, bargap);
58+
59+
%---------------------------------------------------------------------%
60+
61+
%-reformat Z according to offsets-%
62+
m = 1;
63+
lz2 = 0.5*length(Z);
64+
65+
for n = 1:4:lz2
66+
Z(n:n+3) = Z(n:n+3)+offsets(m);
67+
Z(n+lz2:n+lz2+3) = Z(n+lz2:n+lz2+3)+offsets(m);
68+
m = m + 1;
69+
end
70+
71+
%-------------------------------------------------------------------------%
72+
73+
%-set mesh3d data-%
74+
obj.data{surfaceIndex}.x = X;
75+
obj.data{surfaceIndex}.y = Y;
76+
obj.data{surfaceIndex}.z = Z;
77+
obj.data{surfaceIndex}.i = int16(I-1);
78+
obj.data{surfaceIndex}.j = int16(J-1);
79+
obj.data{surfaceIndex}.k = int16(K-1);
80+
81+
%-------------------------------------------------------------------------%
82+
83+
%-coloring-%
84+
cmap = figure_data.Colormap;
85+
86+
if isnumeric(bar_data.FaceColor)
87+
88+
%-paper_bgcolor-%
89+
col = 255*bar_data.FaceColor;
90+
col = sprintf('rgb(%f,%f,%f)', col);
91+
92+
else
93+
switch bar_data.FaceColor
94+
95+
case 'none'
96+
marker.color = 'rgba(0,0,0,0)';
97+
98+
case {'flat','interp'}
99+
100+
switch bar_data.CDataMapping
101+
102+
case 'scaled'
103+
capCD = max(min(cdata(1,1),axis_data.CLim(2)),axis_data.CLim(1));
104+
scalefactor = (capCD - axis_data.CLim(1))/diff(axis_data.CLim);
105+
col = 255*(cmap(1+ floor(scalefactor*(length(cmap)-1)),:));
106+
case 'direct'
107+
col = 255*(cmap(cdata(1,1),:));
108+
109+
end
110+
111+
col = sprintf('rgb(%f,%f,%f)', col);
112+
113+
case 'auto'
114+
col = 'rgb(0,113.985,188.955)';
115+
end
116+
end
117+
118+
obj.data{surfaceIndex}.color = col;
119+
120+
%-------------------------------------------------------------------------%
121+
122+
%-some settings-%
123+
obj.data{surfaceIndex}.contour.show = true;
124+
obj.data{surfaceIndex}.contour.width = 6;
125+
obj.data{surfaceIndex}.contour.color='rgb(0,0,0)';
126+
obj.data{surfaceIndex}.flatshading = false;
127+
128+
%-------------------------------------------------------------------------%
129+
130+
%-lighting settings-%
131+
obj.data{surfaceIndex}.lighting.diffuse = 0.8;
132+
obj.data{surfaceIndex}.lighting.ambient = 0.65;
133+
obj.data{surfaceIndex}.lighting.specular = 1.42;
134+
obj.data{surfaceIndex}.lighting.roughness = 0.52;
135+
obj.data{surfaceIndex}.lighting.fresnel = 0.2;
136+
obj.data{surfaceIndex}.lighting.vertexnormalsepsilon = 1e-12;
137+
obj.data{surfaceIndex}.lighting.facenormalsepsilon = 1e-6;
138+
139+
obj.data{surfaceIndex}.lightposition.x = 0;
140+
obj.data{surfaceIndex}.lightposition.y = 0;
141+
obj.data{surfaceIndex}.lightposition.z = 0;
142+
143+
%-------------------------------------------------------------------------%
144+
145+
%-surface name-%
146+
obj.data{surfaceIndex}.name = bar_data.DisplayName;
147+
148+
%-------------------------------------------------------------------------%
149+
150+
%-surface visible-%
151+
obj.data{surfaceIndex}.visible = strcmp(bar_data.Visible,'on');
152+
153+
%-------------------------------------------------------------------------%
154+
155+
leg = get(bar_data.Annotation);
156+
legInfo = get(leg.LegendInformation);
157+
158+
switch legInfo.IconDisplayStyle
159+
case 'on'
160+
showleg = true;
161+
case 'off'
162+
showleg = false;
163+
end
164+
165+
obj.data{surfaceIndex}.showlegend = showleg;
166+
167+
%-------------------------------------------------------------------------%
168+
169+
%-SETTING SCENE-%
170+
171+
%-------------------------------------------------------------------------%
172+
173+
%-aspect ratio-%
174+
ar = obj.PlotOptions.AspectRatio;
175+
176+
if ~isempty(ar)
177+
if ischar(ar)
178+
scene.aspectmode = ar;
179+
elseif isvector(ar) && length(ar) == 3
180+
xar = ar(1);
181+
yar = ar(2);
182+
zar = ar(3);
183+
end
184+
else
185+
186+
%-define as default-%
187+
xar = max(xedges(:));
188+
yar = max(yedges(:));
189+
zar = 0.7*max([xar, yar]);
190+
end
191+
192+
scene.aspectratio.x = xar;
193+
scene.aspectratio.y = yar;
194+
scene.aspectratio.z = zar;
195+
196+
%-------------------------------------------------------------------------%
197+
198+
%-camera eye-%
199+
ey = obj.PlotOptions.CameraEye;
200+
201+
if ~isempty(ey)
202+
if isvector(ey) && length(ey) == 3
203+
scene.camera.eye.x = ey(1);
204+
scene.camera.eye.y = ey(2);
205+
scene.camera.eye.z = ey(3);
206+
end
207+
else
208+
209+
%-define as default-%
210+
xey = xar; if xey>0 xfac = 5; else xfac = 0.5*length(values); end
211+
yey = yar; if yey>0 yfac = 0.1; else yfac = -0.1; end
212+
if zar>0 zfac = -0.05; else zfac = 0.1; end
213+
214+
scene.camera.eye.x = xey + 7;
215+
scene.camera.eye.y = yey - 2;
216+
scene.camera.eye.z = zar + 0.5;
217+
end
218+
219+
%-------------------------------------------------------------------------%
220+
221+
%-axis configuration-%
222+
scene.xaxis.range = axis_data.XLim(end:-1:1);
223+
scene.yaxis.range = axis_data.YLim;
224+
scene.zaxis.range = axis_data.ZLim;
225+
226+
scene.xaxis.tickvals = axis_data.XTick;
227+
scene.xaxis.ticktext = axis_data.XTickLabel;
228+
229+
scene.yaxis.tickvals = axis_data.YTick;
230+
scene.yaxis.ticktext = axis_data.YTickLabel;
231+
232+
scene.zaxis.tickvals = axis_data.ZTick;
233+
scene.zaxis.ticktext = axis_data.ZTickLabel;
234+
235+
scene.yaxis.zeroline = false;
236+
scene.yaxis.zeroline = false;
237+
scene.zaxis.zeroline = false;
238+
239+
scene.xaxis.showline = true;
240+
scene.yaxis.showline = true;
241+
scene.zaxis.showline = true;
242+
243+
scene.xaxis.tickcolor = 'rgba(0,0,0,1)';
244+
scene.yaxis.tickcolor = 'rgba(0,0,0,1)';
245+
scene.zaxis.tickcolor = 'rgba(0,0,0,1)';
246+
247+
scene.xaxis.ticklabelposition = 'outside';
248+
scene.yaxis.ticklabelposition = 'outside';
249+
scene.zaxis.ticklabelposition = 'outside';
250+
251+
scene.xaxis.title = axis_data.XLabel.String;
252+
scene.yaxis.title = axis_data.YLabel.String;
253+
scene.zaxis.title = axis_data.ZLabel.String;
254+
255+
%-------------------------------------------------------------------------%
256+
257+
%-SET SCENE TO LAYOUT-%
258+
obj.layout = setfield(obj.layout, sprintf('scene%d', xsource), scene);
259+
260+
%-------------------------------------------------------------------------%
261+
262+
end
263+
264+
function bar_ = bar_data(position3d, size_)
265+
% position3d - 3-list or array of shape (3,) that represents the point of coords (x, y, 0), where a bar is placed
266+
% size = a 3-tuple whose elements are used to scale a unit cube to get a paralelipipedic bar
267+
% returns - an array of shape(8,3) representing the 8 vertices of a bar at position3d
268+
269+
if nargin < 2
270+
size_ = [1, 1, 1];
271+
end
272+
273+
bar_ = [...
274+
0, 0, 0; ...
275+
1, 0, 0; ...
276+
1, 1, 0; ...
277+
0, 1, 0; ...
278+
0, 0, 1; ...
279+
1, 0, 1; ...
280+
1, 1, 1; ...
281+
0, 1, 1 ...
282+
]; % the vertices of the unit cube
283+
284+
for n =1:size(bar_, 1)
285+
bar_(n,:) = bar_(n,:) .* size_; % scale the cube to get the vertices of a parallelipipedic bar_
286+
end
287+
288+
289+
bar_ = bar_ + position3d; %translate each bar_ on the directio OP, with P=position3d
290+
end
291+
292+
function [vertices, I, J, K] = triangulate_bar_faces(positions, sizes)
293+
% positions - array of shape (N, 3) that contains all positions in the plane z=0, where a histogram bar is placed
294+
% sizes - array of shape (N,3); each row represents the sizes to scale a unit cube to get a bar
295+
% returns the array of unique vertices, and the lists i, j, k to be used in instantiating the go.Mesh3d class
296+
297+
if nargin < 2
298+
sizes = ones(size(positions,1), 3); %[(1,1,1)]*len(positions)
299+
else
300+
sizes;
301+
% if isinstance(sizes, (list, np.ndarray)) and len(sizes) != len(positions):
302+
% raise ValueError('Your positions and sizes lists/arrays do not have the same length')
303+
end
304+
305+
c = 1;
306+
for n = 1:size(positions, 1)
307+
if sizes(n, 3) ~= 0
308+
all_bars(:,:,c) = bar_data(positions(n,:), sizes(n,:))';
309+
c = c+1;
310+
end
311+
end
312+
313+
% all_bars = [bar_data(pos, size) for pos, size in zip(positions, sizes) if size[2]!=0]
314+
[r, q, p] = size(all_bars);
315+
316+
% extract unique vertices from the list of all bar vertices
317+
all_bars = reshape(all_bars, [r, p*q])';
318+
[vertices, ~, ixr] = unique(all_bars, 'rows');
319+
320+
%for each bar, derive the sublists of indices i, j, k assocated to its chosen triangulation
321+
I = [];
322+
J = [];
323+
K = [];
324+
325+
for k = 0:p-1
326+
aux = ixr([1+8*k, 1+8*k+2,1+8*k, 1+8*k+5,1+8*k, 1+8*k+7, 1+8*k+5, 1+8*k+2, 1+8*k+3, 1+8*k+6, 1+8*k+7, 1+8*k+5]);
327+
I = [ I; aux(:)];
328+
aux = ixr([1+8*k+1, 1+8*k+3, 1+8*k+4, 1+8*k+1, 1+8*k+3, 1+8*k+4, 1+8*k+1, 1+8*k+6, 1+8*k+7, 1+8*k+2, 1+8*k+4, 1+8*k+6]);
329+
J = [ J; aux(:)];
330+
aux = ixr([1+8*k+2, 1+8*k, 1+8*k+5, 1+8*k, 1+8*k+7, 1+8*k, 1+8*k+2, 1+8*k+5, 1+8*k+6, 1+8*k+3, 1+8*k+5, 1+8*k+7]);
331+
K = [ K; aux(:)];
332+
end
333+
334+
end
335+
336+
function [X, Y, Z, I, J, K] = get_plotly_mesh3d(xedges, yedges, values, bargap)
337+
% x, y- array-like of shape (n,), defining the x, and y-ccordinates of data set for which we plot a 3d hist
338+
339+
xsize = xedges(2)-xedges(1)-bargap;
340+
ysize = yedges(2)-yedges(1)-bargap;
341+
[xe, ye]= meshgrid(xedges(1:end-1), yedges(1:end-1));
342+
ze = zeros(size(xe));
343+
344+
positions = zeros([size(xe), 3]);
345+
positions(:,:,1) = xe;
346+
positions(:,:,2) = ye;
347+
positions(:,:,3) = ze;
348+
349+
[m, n, p] = size(positions);
350+
positions = reshape(positions, [m*n, p]);
351+
352+
h = values'; h = h(:);
353+
sizes = [];
354+
for n = 1:length(h)
355+
sizes = [sizes; ysize, ysize, h(n)];
356+
end
357+
358+
[vertices, I, J, K] = triangulate_bar_faces(positions, sizes);
359+
X = vertices(:,1);
360+
Y = vertices(:,2);
361+
Z = vertices(:,3);
362+
363+
end

0 commit comments

Comments
 (0)