VENN Plot 2- or 3- circle area-proportional Venn diagram

VENN offers the following advantages over the VennX function available on the FEX:

1) It's much faster

2) Draws venn diagram as patch objects, allowing much greater flexibility in presentation (edge/face colors, transparency, etc.)

3) Returns a structure with required plotting data, allowing the user to make additional modifications (translation, scaling, rotation, etc.)

4) Supports multiple methods for optimizing the position of three-circle venn diagrams.

venn(A, I)

venn(Z)

venn(..., 'ErrMinMode', MODE)

H = venn(...)

[H, S] = venn(...)

[H, S] = venn(..., 'Plot', 'off')

S = venn(..., 'Plot', 'off')

[...] = venn(..., P1, V1, P2, V2, ...)

venn(A, I) by itself plots circles with total areas A, and intersection area(s) I. For two-circle venn diagrams, A is a two element vector of circle areas [c1 c2] and I is a scalar specifying the area of intersection between them. For three-circle venn diagrams, A is a three element vector [c1 c2 c3], and I is a four element vector [i12 i13 i23 i123], specifiying the two-circle intersection areas i12, i13, i23, and the three-circle intersection i123.

venn(..., 'ErrMinMode', MODE)

Used for 3-circle venn diagrams only. MODE can be 'TotalError' (default), 'None', or 'ChowRodgers'. When ErrMinMode is 'None', the positions and sizes of the three circles are fixed by their pairwise-intersections, which means there may be a large amount of error in the area of the three-circle intersection. Specifying ErrMinMode as 'TotalError' attempts to minimize the total error in all four intersection zones. The areas of the three circles are kept constant in proportion to their populations. The 'ChowRodgers' mode uses the the method proposed by Chow and Rodgers [Ref. 1] to draw 'nice' three-circle venn diagrams which appear more visually representative of the desired areas, although the actual areas of the circles are allowed to deviate from requested values.

H = venn(...) returns a two- or three- element vector of handles to the patches representing the circles.

[H, S] = venn(...) returns a structure containing descriptive valuescomputed for the requested venn diagram.

[...] = venn(..., P1, V1, P2, V2, ...) Specifies additional patch settings in standard Matlab parameter/value pair syntax. Parameters can be any valid patch parameter. Values for patch parameters can either be single values, or a cell array of length LENGTH(A), in which case each value in the cell array is applied to the corresponding circle in A.

See venn.m for additional options, descriptions, references, and examples.

Darik (2021). venn (https://www.mathworks.com/matlabcentral/fileexchange/22282-venn), MATLAB Central File Exchange. Retrieved .

Created with
R2008b

Compatible with any release

**Inspired by:**
Proportional Venn Diagrams

Find the treasures in MATLAB Central and discover how the community can help you!

Start Hunting!Create scripts with code, output, and formatted text in a single executable document.

Florian SchwaigerFixed the bug "Triangle inequality not satisfied". Replace the block raising the error by the following code. Reason is, if one circle is within the other, all three circles are on one line and due to minimization inaccuracy, d1+d2<d3 is possible, violating the triangle equation.

%Check triangle inequality

srtD = sort(d);

if srtD(end) < (srtD(1) + srtD(2))

%Guess the initial position of the third circle using the law of cosines

alpha = acos( (d(1)^2 + d(2)^2 - d(3)^2) / (2 * d(1) * d(2)) );

x(3) = d(2)*cos(alpha);

y(3) = d(2)*sin(alpha);

else

% Guess the initial position of the third circle using mean

% distance

IM = max(I0);

if (I0(2) == IM && A0(1) > A0(3)) || (I0(3) == IM && A0(2) > A0(3))

% #1 is the inner most circle

x(3) = (d(1) + d(2) - d(3)) / 2;

elseif (I0(1) == IM && A0(2) > A0(1)) || (I0(2) == IM && A0(3) > A0(1))

% #2 is the inner most circle

x(3) = (d(1) - d(2) - d(3)) / 2;

elseif (I0(1) == IM && A0(1) > A0(2)) || (I0(3) == IM && A0(3) > A0(2))

% #3 is the inner most circle

x(3) = (d(1) + d(2) + d(3)) / 2;

end

end

cmoMatthewTong ShengWith regards to "When I put text onto the graph (suppose for simplicity I want to put the amount % of each zone in each zone centroid), when I set a transparency (alpha <1), the text becomes faded as if the circles were covering it."

Just to update, this is no longer an issue when I export the figure as jpg, etc.; the text appears opaque and not covered by the patches.

Tong ShengThank you, Darik. I will give that a try.

Another comment/question: when I put text onto the graph (suppose for simplicity I want to put the amount % of each zone in each zone centroid), when I set a transparency (alpha <1), the text becomes faded as if the circles were covering it. Ideally I would like the text to be bold and clear on top of the semi-transparent circles. Anyone else have this same problem, or know of a workaround? Thanks.

Darik@Tong: I suppose you could tweak the position of the circles manually via the returned handles:

%Build the venn plot and return the handles

h = venn(...)

%Shift the first circle by (dx, dy)

set(h(1), 'XData', dx + get(h(1), 'XData'))

set(h(1), 'YData', dy + get(h(1), 'YData'))

Tong ShengI have a 3-circle diagram in which one of my zones is empty (for [z1 z2 z3 z12 z13 z23 z123], z1 = 0), but an area/patch is still drawn. I think if the centroid for circle1 is shifted so that it fits completely inside the other two circles, that would be ok. How can I do this?

Fabiano Baronigreat work, thanks.

I noticed that sometimes this error is returned: "Triangle inequality not satisfied", for example if A=[15 14 3]; I=[13 1 0 0].

It seems to be related to the constrain that circle centres shall not be further apart than rA+rB:

D = fminbnd(@areadiff, 0, rA+rB, opts);

Can you please suggest a solution or a workaround for this issue? Many thanks.

HirenEugene GallagherThanks Darik. You were right. My 2013a had a faulty Matlab path statement and wasn't finding my optimization toolbox. I changed back to the default path, rebuilt my path for my directories, and Venn.m worked beautifully. Thanks.

DarikEugene: optimset is a function in the optimization toolbox. I suspect you have that toolbox installed in your Matlab 2012, but not 2013.

Eugene GallagherWhile the function works great in Matlab Release 2012a, it produces errors in Release 2013a:

>> figure, axis equal, axis off;A=[100 100];I=25;venn(A,I,'FaceColor',{'r','y'},'FaceAlpha',{1,0.6},'EdgeColor','black')

Undefined function 'optimset' for input arguments of type 'char'.

Error in venn>parseArgsIn (line 860)

fminOpts = [optimset('fminbnd'), optimset('fminsearch')];

Error in venn (line 166)

[A0, I0, Z0, nCirc, fminOpts, vennOpts, patchOpts] = parseArgsIn (varargin);

Darik@RyanG sure, just set the intersection between those two circles equal to the area of the smaller (enclosed) circle. But depending on the area of the third circle and its intesections, a solution may not be possible.

Ryan GIs it possible to create a 3-circle venn diagram such that 1 circle is entirely inside another?

Darikthe cyclist: that would be a pretty embarrassing bug on my part. what about circles drawn by other functions, like:

cla, rectangle('Curvature', [1 1], 'Position', [0 0 1 1]); axis equal

?

the cyclistGreat function. Seems to work just as advertised (in my limited trial). However, the areas don't look quite circular to me, whether or not I apply "axis square" after running venn.

Is that expected? I am using R2012b.

Rob CampbellMattmoggces[h, s] = venn([49, 58, 108], [33, 1, 17, 0], 'ErrMinMode', 'ChowRodgers');

This will prompt errors and can't generate h and s variables.

JeremyNevermind my last comment. As long as you center the text labels vertically and horizontally, the zone centroids work quite well. Nice work!

JeremyGreat job on the function. My only beef is that the zone centroids don't seem to be quite accurate.

DarikHi Yuri,

You can try the ChowRodgers mode, which adjusts area ratios instead of absolute areas, and seems to give more satisfying results when actual solutions are impossible. Unfortunately when I tried your dataset I found a small bug in the code -- I just submitted a fix, but it will take a day or two to show up on FEX. If you want you can just fix it yourself: line 563 says:

x = NaN; y = NaN;

It should be edited to include:

x = NaN; y = NaN; sinTp = NaN; cosTp = NaN;

Then you can plot the approximate venn diagram with

venn([477 487 370 3 54 5 2], 'ErrMinMode', 'ChowRodgers')

Yuri KTrying to build venn diagram with the vector v=[477 487 370 3 54 5 2];

Without any additional parameters I've got an error:

Exiting: Maximum number of function evaluations has been exceeded

- increase MaxFunEvals option.

Current function value: Inf

Worked only with

venn(v,'ErrMinMode','None');

but didn't get any 123 intersection. No problem with vennX, but I really like the idea of vector graphics.

I don't need exact proportions, something close would be good enough.

OmarEran MukamelThanks for the bug fix!

DarikHey Eran, can you paste the error you get? Also what version of Matlab are you using?

Eran MukamelFunctionality of syntax: [H, S] = venn(...) does not work for me.

Or ZukDarikHi Or,

You can do this with the returned structure:

[H, S] = venn(...)

for i = 1:length(S.ZoneCentroid)

text(S.ZoneCentroid(i,1), S.ZoneCentroid(i,2), num2str(S.ZonePop(i)));

end

Or ZukWorks great.

The only thing missing (which appears in VennX) is a display

the sizes of each set on the venn diagram itself. Is it easy to add this?