How to resize figure without moving contents

조회 수: 11 (최근 30일)
Alexander Laut
Alexander Laut 2018년 10월 9일
편집: Alexander Laut 2018년 10월 11일
To standardized my figure export process I use a custom defined printfigs.m call at the end of each script. One of the key things it does is timestamp and add a path to each figure at the top so that I can have a record of which script generated which figure.
To prevent this from overlapping any title's, annotations, or other parts of the figure, I want to resize the figure by adding a margin to the top that I can then use to place my annotation without potentially obstructing anything in the figure. I want a code that can take any figure and stretch the margins without disturbing it's contents.
To accomplish this, I've written this little code which I've been using for the bast two years on 2016b with little issue, but now it fails in 2018b quite frequently.
function [] = resizeFig(t, l, b, r)
fh = gcf();
set(findall(fh,'-property','Units'),'Units', 'pixels'); %%Set Object Sizes to Pixels
set(fh,'position',get(fh,'position')+[0,0,l+r,t+b]); % extends range of figure only
%%Grab Re-Sizeable Objects
objs = findobj(fh,'-property','position'); % grabs all objects with position properties
oPos = get(objs,'position'); % grabs position of moveable objects
ind = cellfun(@(C) size(C,2)==4,oPos); % finds objects that take 4 vector position input
objs = objs(ind);
oPos = oPos(ind);
%%Resize Objects within Figure
nPos = cellfun(@(C) C+[l,b,0,0],oPos,'uniformoutput',false); % displace positions left and down
for i = 1:length(objs)
set(objs(i),'position',nPos{i});%+[dleft,dbot,0,0]);
end
%%Set objects back to normalized/rescaleable
set(findall(fh,'-property','Units'),'Units', 'normalized');
end
Now I do hope there is a smarter way to do this to improve performance, but more importantly the line:
oPos = get(obs,'position');
tends to fail with this error:
Error using matlab.graphics.Graphics/get
No public property 'Position' for class
'ToolbarStateButton'.
this surprises me since the previous line is purposefully chosen to define objs as only those with the position property, though it now complains that it doesn't have that property. The same occurs if I use findall instead of findobj.
What has changed since 2016b and 2018b to cause this to fail?

채택된 답변

Greg
Greg 2018년 10월 10일
편집: Greg 2018년 10월 10일
Two things are actually happening. The first is somewhat nitpicky: there is no public property Position for the ToolbarStateButton. The use of findobj is allowed to see the existence of the (let's assume private) property Position, but your use of set is not allowed because it must be public to be set.
The second thing that happened is R2018b introduced some improved (?) (I think so, but you likely disagree right now) axes interaction default functionality. Most of the simple interactions are now ToolbarStateButton objects in a fancy floating-and-auto-appearing axes toolbar, instead of static buttons in the figure toolbar. These new ToolbarStateButton objects apparently have the non-public Position property.
Note: In my tests, findobj did not return the ToolbarStateButton components. I had to use findall.
To fix your code, simply delete the toolbar.
addToolbarExplorationButtons(fh); % Put the old menu buttons back just in case you need them
delete(ah.Toolbar); % ah is an axes handle in the figure
findobj(...);
  댓글 수: 3
Greg
Greg 2018년 10월 10일
I suspect DefaultFigureToolbar has nothing to do with it, but haven't run any tests to confirm.
There are lots of different possible approaches to this task.
  • If you're running it programmatically in line with figure generation, you could set the figure visibility to 'off' to prevent user interaction.
  • You could stick to top or right margins whereby the left and bottom of each component are good as-is after changing to units other than normalized.
  • You could create a new figure, put a panel of the original figure's size in it, and copy components into the panel.
  • Probably a lot of other really bad ideas, but they would technically do what you're attempting.
Alexander Laut
Alexander Laut 2018년 10월 11일
편집: Alexander Laut 2018년 10월 11일
I think I found my issue, in my code elsewhere I was calling:
set(0, 'ShowHiddenHandles', 'on')
I think I wrote this into one of my codes like 3 years ago as I was playing around with hiding figures until my code was finished so my printfigs function would by default show them all before export to png.
Heres a test code I wrote that should clarify my issue:
clc;clear variables; close all
% Default Setup (Works; with or without toolbar)
% set(0,'ShowHiddenHandles','off');
% set(0,'defaultfiguretoolbar','auto');
% Alternative Setup (Works; no toolbar)
% set(0,'ShowHiddenHandles','off');
% set(0,'defaultfiguretoolbar','none');
% Alternative Setup (Fails; when toolbar initiated)
set(0,'ShowHiddenHandles','on');
set(0,'defaultfiguretoolbar','auto');
% Generate Test Scenario
figure();
axis();
fprintf('try hovering over figure with mouse to initiate toolbar...\n')
pause(3)
objs = findobj(gcf(),'-property','position'); % grabs objects with position
objs_pos = get(objs,'position'); % grabs positions of objects
disp('Completed without error!')

댓글을 달려면 로그인하십시오.

추가 답변 (0개)

카테고리

Help CenterFile Exchange에서 Specifying Target for Graphics Output에 대해 자세히 알아보기

Community Treasure Hunt

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

Start Hunting!

Translated by