-
Notifications
You must be signed in to change notification settings - Fork 232
Scaling for math content in a block element:
Hello there,
I have a question regarding a Scaling possibility for math content in a block element:
the class .MathJax_Display is assigned to any block element containing math with the style width:100% - therefore it takes the width value of the parent element, in my case a sidebar with width 400px size.
Sometimes there are equations and formulas which are slightly bigger in width and therefore they overlap.
Now i want to reduze the font-size a bit to decrease the width in a way that the equation is 400px.
First question:
is there a way to achieve this result within mathjax by configuring the scale property dynamically?
And second:
Is it possible to remove the style="width:100%" property to get the real width of the generated element? (I could than start manipulating the width by reducing the font size and checking via jquery if it already fits.)
I already tried it this way:
http://groups.google.com/group/mathjax-users/browse_thread/thread/662c469d08c59260
but with no luck.
Thanks for your help and best regards stephan c.
Well, one night and here is a (a very dirty hack, but I guess it will work until an other method is provided)
The
.MathJax_Display divs get automatically the css attributes width:100%; and position:relative; I removed this by editing the jax/output/HTML- CSS/jax.js (Looking for the second appearance of .MathJax_Display) Afterwards I added the display:table; shrinkwrapping method to .MathJax_Display in the default.js:
styles: { ".MathJax_Display": { display:"table" }
No I can catch the actual width of the generated math by jquery via:
jQuery('div.MathJax_Display').each(function(i) { var currentwidth = jQuery(this).width(); });
Hope that helps somebody! stephan c.
Here is an example that implements what I think you are looking for:
<script type="text/javascript" src="http://cdn.mathjax.org/mathjax/2.0-beta/MathJax.js?config=TeX-AMS_HTML "> </script> <div id="dynamic" style="margin:1em 0; text-align:center">
\(\displaystyle x+x+x+x+x+x+x+x+x+x+x+x+x+x \)
</div>
</div>
<script>
MathJax.Hub.Queue(function () {
var math = document.getElementById("dynamic");
math.style.display = "inline"; var w = math.offsetWidth;
math.style.display = ""; if (w > 400) {math.style.fontSize = Math.floor(400/w * 100)+"%"} MathJax.Hub.Queue(["Rerender",MathJax.Hub,math]); }); </script>
No modifications are needed to MathJax, and you do not alter the
styles for any of MathJax's output. Instead, you use your own DIV
around the math to be adjusted (you can use a class to identify this
if you want, an use jQuery as in your own example. In order to get
the true width of the math, you use in-line mode with \displaystyle
instead. Then in your routine that determines the width, you
temporarily change the display CSS to inline (so it gets the natural
width) and then put it back to block after getting the width. If the
width is bigger than the container, you scale your DIV appropriately
and then re-render the mathematics using that scale (this step is not
absolutely required, but the results will be better if you re-render
it).
Note that you will need to have a container whose font-size you can
change (since each equation has its own scaling) and you can't use the
one produced by MathJax, since that will be removed and replaced by
the re-rendering. So using the extra DIV is really necessary and not
extra after all. The styling on the div is to replace the margin and
centering that the displayed equation would have had, since we use \
(\displaystyle ... ) rather than [...].
Finally, this approach will work no matter what rendering mode the
user has chosen, whereas solutions based on the MathJax_Display
element will only work with HTML-CSS output (the other renderers use
different classes for their output). It is best not to hack on the
HTML-CSS output itself, since that may change from version to version,
and it makes your results not work in NativeMML or SVG output.
In any case, this should point you in the right direction.
Davide
It is better not to edit the HTML-CSS jax.js file directly, but rather
use your configuration to modify it (people using the CDN don't have
access to do that anyway). For example
<script type="text/x-mathjax-config">
MathJax.Hub.Config({
"HTML-CSS": {
styles: {
".MathJax_Display": {
width: null,
position: null,
display: "table"
}
}
}
});
</script>
would remove the width and position values and add the display:table
value that you want.
I don't recommend making this change, however, as this will affect not
only the math in the sidebar, but ALL displayed math. I'm not certain
what the effect of setting display:table will be in general.
Finally, your approach will only work for HTML-CSS output, not the
NativeMML or SVG output, and so if your user selects a different
renderer (or if the configuration chooses between NativeMML and HTML-
CSS (as most do), then you won't get the rescaling that you want.
So I recommend the example I sent earlier as a better method of
handling this.
Davide
Hello Davide,
thanks for your feedback - you are right, modifiying the the js files is bad so I use know the second version - for the first Version I will need a bit more time to implement.
display:table is on of the easiest ways do "shrinkfit" a div so it will not take the width of its parent:
http://haslayout.net/css-tuts/CSS-Shrink-Wrap
stephan
I put together a test page demonstrating the simplest version of what Davide described -- here is the link:
http://cs.jsu.edu/~leathrum/mjscaletest.html
If you view page source on this, you should be able to see everything you need about using the Rerender call and getting height and width data. The key to this is to use:
\ (\displaystyle ... )<\span>
I put a thin solid red border on the span to highlight what it is doing. I used the CSS display:inline-block setting instead of display:table -- the default display:inline didn't manage the height of the span correctly for this purpose. You also need to have a non- null initial width setting, so width:auto gets things started. The height and width data don't show up until you click a button.
Note that this demonstrates getting height and width data -- what you do with it after that is up to you. ;-)
Hey, thanks for the example! I will implement it as soon as I can and will give feedback if something is still vague to me - but I think you nailed it with this example :) best regards stephan
Here is a modified version of your page.
<!DOCTYPE html>
<html>
<head>
<script type="text/javascript"
src="http://cdn.mathjax.org/mathjax/latest/MathJax.js?config=TeX-AMS-MML_HTMLorMML
"> </script> <script> function changeSize(button) { var myeqn = document.getElementById('myeqn'); myeqn.style.fontSize = button.textContent; MathJax.Hub.Queue( ['Rerender',MathJax.Hub,'myeqn'], function () { document.getElementById('mylabel').innerHTML = 'width: '+myeqn.offsetWidth+", height: "+myeqn.offsetHeight; } ); } </script>
Font size: x-small small medium large x-large <span id="myeqn" style="border:thin solid red; display:inline-
block;width:auto;"> (\displaystyle{\int_{-\infty}^\infty e^{-x^2}, dx=\sqrt{\pi}})
</body>
</html>
I have put the re-rendering and measuring into one function called by
all the buttons. The key change, however, is that the looking up of
the size really should be part of the queued functions as well, since
the rendering may be asynchronous. Since the math has already been
rendered once, it is probably true that everything has already been
loaded, but it might be that a stretchy delimiter (for example) is at
a different size when the math is scaled, and that might require
loading a font, which is asynchronous. Or it might be that some other
piece of code is listening on the "New Math" signal that will be
generated when the math is rerendered, and that IT starts an
asynchronous action that you have to wait for. So you really do want
to have the size lookup in the queue as well so that you are sure it
isn't done until after the math is ready.
The inline-block is also a good choice. I had used inline because
only the width was of concern, but inline-block does give you both, as
you point out.
Davide
OK, page at my link is updated with your code Davide. Good point about pushing the width/height stuff onto the queue, too. Also clever of you just to use the button's text-content to specify the size, since I was using the same label anyway.
I am noticing that in some contexts, the browser doesn't seem to show much difference between "medium" and "large" (FF10 in Ubuntu I see normal-looking changes, FF10 in Fedora there is no change in height or width and only minor changes in math layout). Is this just because of availability of certain font sizes or something like that?
MathJax tries to match the ex-hight if its fonts to that of the
surrounding font, but it only has pixel-level resolution, so it may be
that MathJax can't resolve the difference between those two sizes in
the default font for your browser. You could try setting
<body style="font-size:200%">
and see if you get a better difference that way.
Davide
You might try making a div that contains the span and also some text,
and set its font-size (just using the inner span to measure the
mathematics) and see if the text is getting scaled properly (and to
see how well MathJax is matching the text).
Davide
You could try setting
and see if you get a better difference that way.
This made the rest of the text double size, and the initial MathJax display double size, but as soon as I clicked the buttons (which use the "small", "medium", "large", etc. sizes, and which are absolute) the MathJax went back to the sizes I was seeing before -- still not much difference between "medium" and "large".
You might try making a div that contains the span and also some text, and set its font-size
This was a bit more revealing: when I did this, the surrounding text behaved similarly to the MathJax -- the "large" size text font was heavier and a bit wider than the "medium" size, but not significantly taller. I'm sure MathJax is trying to find its size context from a height measurement (since that is the normal way to measure a font), so this would explain why the MathJax isn't changing much. It must be something about the default font settings for this browser, but I couldn't find anything unusual in the preferences.... Nothing that affects MathJax as a whole, to be sure.
You could try setting
and see if you get a better difference that way.This made the rest of the text double size, and the initial MathJax display double size, but as soon as I clicked the buttons (which use the "small", "medium", "large", etc. sizes, and which are absolute) the MathJax went back to the sizes I was seeing before -- still not much difference between "medium" and "large". OK, I thought they were relative, but I guess not (at least not in
some browsers).
You might try making a div that contains the span and also some text, and set its font-size
This was a bit more revealing: when I did this, the surrounding text behaved similarly to the MathJax -- the "large" size text font was heavier and a bit wider than the "medium" size, but not significantly taller. I'm sure MathJax is trying to find its size context from a height measurement (since that is the normal way to measure a font), so this would explain why the MathJax isn't changing much. It must be something about the default font settings for this browser, but I couldn't find anything unusual in the preferences.... Nothing that affects MathJax as a whole, to be sure. Yes, MathJax tries to match the ex-height of its fonts with that of
the surrounding font. So if the surrounding fonts aren't changing
much, neither will MathJax's.
Davide