Efficient way to draw a patch cube with each half a different colour

55 Views Asked by At

I need to draw a cube using the patch() command, where the cube is one colour for y>0 and another colour for y<0. Something like this:

enter image description here

I would ideally like the user to be able to simply call patch with a single command (rather than looping over the patch command), like this

patch(x_coords, y_coords, z_coords, colour_data)

I made a working version, as shown in the following, but can anyone suggest a more succinct way to do this?

%%% TOP SECTION VERTICES %%%
xf1 = [1 1 1 1].';   % Face 1
xf2 = [1 1 -1 -1].'; % Face 2
xf3 = [1 1 -1 -1].'; % Face 3
xf4 = -xf1;          % Face 4
xf5 = xf2;           % Face 5

yf1 = [0 1 1 0].';
yf2 = [0 1 1 0].';
yf3 = [1 1 1 1].';
yf4 = yf1;
yf5 = yf2;

zf1 = [1 1 -1 -1].';
zf2 = [1 1 1 1].';
zf3 = [-1 1 1 -1].';
zf4 = zf1;
zf5 = -zf2;

x_half1 = [xf1 xf2 xf3 xf4 xf5];
y_half1 = [yf1 yf2 yf3 yf4 yf5];
z_half1 = [zf1 zf2 zf3 zf4 zf5];

%%% BOTTOM SECTION VERTICES
x_half2 = x_half1;
y_half2 = -y_half1;
z_half2 = z_half1;
%%%

x_verts = [x_half1 x_half2];
y_verts = [y_half1 y_half2];
z_verts = [z_half1 z_half2];

% Define patch colours
c1 = [255 0 0]/255; % Red colour
c2 = [0 0 255]/255; % Blue colour
C = zeros(10,1,3);
C(1:5,1,1) = c1(1); % First x5 faces are colour 1
C(1:5,1,2) = c1(2);
C(1:5,1,3) = c1(3);
C(6:10,1,1) = c2(1); % Next x5 faces are colour 2
C(6:10,1,2) = c2(2);
C(6:10,1,3) = c2(3);

figure;
patch(x_verts,y_verts,z_verts,C)
view(3)
1

There are 1 best solutions below

5
On

You could define the drawCuboid function as defined at the bottom of this question, then just call it twice with different colours for either side of the y axis. The function works out the locations of the 8 vertices of your cube (or cuboid), then assigns them to a matrix in the correct order required by patch to define the 6 faces of 4 vertices. See the code comments for more details.

Usage would look like this:

O     = [-1,0,-1]; % define the vertex 1 origin, note y=0
Lred  = [2,1,2];   % define the red cuboid for positive y
Lblue = [2,-1,2];  % define the blue cuboid for negative y
% Draw two cuboids
figure; hold on; grid on;
drawCuboid( O, Lblue, [0,0,1] );
drawCuboid( O, Lred,  [1,0,0] );
xlim( [-2,2] ); ylim( [-2,2] ); zlim( [-2,2] );

cuboids

You can get the same result in a single function call, automatically splitting on y=0 with the following function:

O     = [-1,-1,-1];
L     = [2,2,2];
drawSplitCuboid( O, L ); 

function drawSplitCuboid( O, L )    
    figure; hold on; grid on;
    if O(2) < 0
        Lblue = L;
        Lblue(2) = min( -O(2), L(2) );
        drawCuboid( O, Lblue, [0,0,1] );
    end
    if O(2) + L(2) > 0
        Lred = L;
        Lred(2) = L(2) - abs(O(2));
        O(2) = max(0,O(2));
        drawCuboid( O, Lred,  [1,0,0] );
    end
    xlim( [-2,2] ); ylim( [-2,2] ); zlim( [-2,2] );
end
function drawCuboid( O, L, C )
    % Draw a cuboid with the first vertex at the origin "O" which spans 
    % the x, y and z axes by distances L(1), L(2) and L(3) respectively.
    % The distances can be negative.
    % The cube will have colour C

    % Define all verticies of the cube as the origin with or without adding
    % the length of each side. All combinations of with/without gives 8
    % vertices. = combvec([0,1],[0,1],[0,1])
    xi = [ 0 1 0 1 0 1 0 1
           0 0 1 1 0 0 1 1
           0 0 0 0 1 1 1 1 ];
    cnr = O(:) + xi.*L(:);

    % From the order of xi above, we define the corners starting from
    % the origin at vertex 1, the front face clockwise is [1,5,6,2]. Then
    % the apex on the back face along the y axis is 3, and the back face
    % clockwise from the same perspective is [3,4,8,7]. With that ordering,
    % we can define all faces (one per column) in terms of the vertices.
    % Order is Front/Top/Right/Back/Left/Bottom
    faces = [ 1 5 2 3 1 1
              2 6 4 4 3 2
              6 8 8 8 7 4
              5 7 6 7 5 3 ];

    % Combine the corner coordinates and apex numbers to define all faces
    X = cnr(1,:); X = X(faces);
    Y = cnr(2,:); Y = Y(faces);
    Z = cnr(3,:); Z = Z(faces);

    % Use patch to draw the cube
    patch( X, Y, Z, C )
    view(3)
end