Settings

Theme

Tips and tricks to write LaTeX papers in with figures generated in Python

github.com

183 points by Wookai 7 years ago · 67 comments

Reader

akshayn 7 years ago

> We also recommend to save the command used to generate a figure in the LaTeX file

An approach I have adopted recently is Knitr[1], so this layer of indirection goes away. With knitr, my data goes directly into the paper repository, and then my Makefile has something like this:

  %.tex: graphs/%.Rnw
    Rscript -e "library(knitr); knit('$?')"
The nice thing is exactly what the authors recommend: it's much easier to enforce a standard appearance across all the figures, and automatically incorporate more recent data into the paper as part of the compilation process.

[1] https://yihui.name/knitr/

abhgh 7 years ago

I'd also add that for figures Inkscape is invaluable [1]. Save as svg once, and export it as whatever later. I typically export it to PDF (from within Inkscape) for pdflatex.

While its typically indispensable for schematics, I often seem to run into the use case of combining previously generated plots or figures, or adding a label/text. Since Inkscape can import pngs, this is a breeze with it. I don't have to go back to the original code to regenerate plots, or fiddle around with latex to make minor adjustments.

For stuff generated via matplotlib, I'd strongly recommend seaborn as an additional library [2]. This is a wrapper over matplotlib. It can prettify plots with just an import and a 'set' command. You can, of course, use it to plot too, and for stuff doable in matplotlib using the seaborn alternative is much easier and looks better with little or no work. And they support pandas dataframes.

[1] https://inkscape.org/

[2] https://seaborn.pydata.org/

  • fourier_mode 7 years ago

    The problem with inkscape is that, any slight changes to the figures would make the user go deep into the workflow pipeline to make the changes. However using LaTeX packages like TikZ or PSTicks would simplify the workflow and make the document more maintainable.

    • abhgh 7 years ago

      I think this is one of those things that depend on your actual workflow, content etc. I see your point; for me this hasn't been a problem.

  • zapnuk 7 years ago

    Have you considered plotly [1] as an alternative to matplotlib?

    It is mostly known for online plots but it has a free offline API that can export plots to eps/pfd/png/etc.

    Parts of their libraries are aimed for interactive plots but imo the basic plots look even better than those of seaborn.

    [1] https://plot.ly/python/

  • radus 7 years ago

    I do this as well. And you can save svg files from matplotlib for editing or composition in Inkscape!

Jill_the_Pill 7 years ago

Having just completed a dissertation in LaTex, with figures online in Overleaf and Dropbox (some of them screenshots), scripts and data spread across two computers and an external hard drive, desperate last minute plot text changes right in the pdf, I just have to ask: WHY DIDN"T YOU POST ALL THIS SOONER?

  • WookaiOP 7 years ago

    I'm sorry ! It has been online for 4 years now, I simply never thought of sharing...

sigurdjs 7 years ago

If you are serious about making beautiful figures in latex, I would seriously recommend using tikz and pgf-plots. It is quite easy to automatically generate tikz-code from python (after all it is supposed to be read and written by humans) and all aspects of the figure can easily be customized. I have been quite successful in generating automated reports with pretty and easily readable figures using tikz and pgf.

If anyone is interested I have uploaded a sample script for generating XY-plots from two numpy lists to github. The code is by no means very good, but I just wanted to share in case anyone wants to try this approach.

https://github.com/sigurdjs/python-tikz

  • programLyrique 7 years ago

    And it's also possible to directly load a csv file with all the data in latex and plot if with pgf, which makes it possible to keep all the plotting options in the latex file:

      \addplot table[x ={Column1}, y ={Column2}] {myData.csv};
    
    The issue is that it can take some time for pgf to load the data and do computations on them, but you can use the external library of tikz so that it does not compute the plot again (and save it as a pdf for later uses).
  • WookaiOP 7 years ago

    Indeed, TikZ is great to create beautiful-looking plots! A friend of mine is quite good at it, e.g. look at Figure 2 in [1] and Figure 2.1 (p. 18) in [2]. For complex figures, though, I find that TikZ can be a bit hard to master and sometimes results in longer compilation times.

    [1] http://www.hrzn.ch/publications/tf-icnp15.pdf [2] http://www.hrzn.ch/publications/thesis.pdf

jedberg 7 years ago

> When writing LaTeX documents, put one sentence per line in your source file.

An interesting tip, never thought of that! It changes the way you write a bit, but it does make finding changes easier, finding errors easier, and forces you to think more about each sentence since you have to hit "enter" at the end of each one.

  • bo1024 7 years ago

    It also works much better with version control software (git). Not only does it help with diffs, as the article mention, but it makes merging way easier in case you and your coauthor change two adjacent sentences at the same time.

  • WookaiOP 7 years ago

    It takes a while to get the "muscle memory" of adding a new line after each sentence but it does make things much easier!

  • u801e 7 years ago

    Interestingly enough, I made a very similar comment[1] about code line length and only using one statement per line in another HN thread several days ago.

    [1] https://news.ycombinator.com/item?id=19349464

bonoboTP 7 years ago

I find it useful to work with plots in Jupyter notebooks. Use the "%matplotlib notebook" cell magic to get interactive plots inline.

Then you can use savefig when it looks good. Then save the code you used into some file near the Latex sources.

mlthoughts2018 7 years ago

I also recommend separating repetitive parts of plot generating code into template files, such as with mako or jinja2, and then programmatically generate sequences of plots by first piping the data into the jinja2 template, and then using insert commands to insert it into a bigger tex document.

I found this helpful when writing a paper where the appendix needed over 35 different tables of regression results, all with the same format but populated with data from different subpopulations, which would need to be regenerated (including updated captions, etc.) any time data cleaning or methodology was changed.

  • WookaiOP 7 years ago

    That's a great point! Templates are a great tool to generate big tables from results, I usually do that for most of the results in my papers, makes it easier to have the odd copy/paste error. I might add this to the tips and tricks, thanks!

unwind 7 years ago

Meta: there seems to be an extra "in" in the title, that makes no sense to me, at least.

Not a native speaker, though.

  • WookaiOP 7 years ago

    Oops, I forgot to erase that when shortening the title, sorry! Not sure I can update it now :(...

  • naniwaduni 7 years ago

    It's a careless transposition of "papers in LaTeX" → "LaTeX papers in" without removing the "in".

euske 7 years ago

Re: figures in EPS. I think SVG is the way to go. It can be generated with matplotlib or even a simpler script (it's just an XML after all). It can be hand edited. It's viewable with a browser. And it can be converted to PDF with rsvg-convert.

I personally find matplotlib a bit unintuitive to use, so I made a 100-line script for generating SVG. It's great.

knolan 7 years ago

This is probably most useful for postgrad students getting started with writing with TeX.

It’s worth pointing out that the figures are made using the matplotlib library, which is primarily based on Matlab’s plotting functionality. This is perhaps just as useful for new researchers as many of them are taught Matlab exclusively throughout their undergraduate courses.

  • p10_user 7 years ago

    It’s great for getting started, but if you start really customizing your plots the Object oriented usage of matplotlib is really the way to go.

jonathanpoulter 7 years ago

A minor plug: I've found I generate graphs and tables in Jupyter notebooks, so I wrote ipynb-tex, to allow you to reference cells from a notebook directly in your LaTeX documents. This supports tables, and figures.

https://github.com/poulter7/ipynb-tex

mychele 7 years ago

I would suggest checking matplotlib2tikz and matlab2tikz to get pgfplot/tikz figures from matplotlib and matlab plots

  • WookaiOP 7 years ago

    Indeed, they're pretty cool (although in my experience the resulting TikZ code sometimes slows down compilation quite a bit).

semi-extrinsic 7 years ago

One itch which (curiously) I can't seem to quite scratch in LaTeX is that it should be possible to say "plot equation \ref{eq:smth} for X in (-4,4)" and just get the bloody graph. Why should I need to define the equation again in a separate place, perhaps even in a separate file?

  • WookaiOP 7 years ago

    It's not exactly what you want, but you can do parametrized plots in TikZ to plot a given function for some range: http://www.texample.net/tikz/examples/parameterized-plots/.

    I wouldn't be surprised if someone wrote a tool that allows you to use a reference to an equation as the function to plot :).

  • dmlorenzetti 7 years ago

    This is not what you asked for, since it still requires a separate file. However it might be close enough to what you want, and -- for complicated expressions -- possibly even better.

    You can write (or derive) the expression using sympy, then have sympy generate a numpy expression that can be evaluated. Sympy can also generate the LaTeX code for any expression. So while that isn't an in-LaTeX solution, it may be close to what you want.

    Johansson's "Numerical Python" shows several examples of this. I will scavenge one of his examples below (trusting it falls under "fair use", and hoping I transcribe it correctly -- note I have left out the imports). The example uses sympy to generate and plot Taylor series expansions of sin(x).

    The key bit to look for in the example is `sympy.lambdify()`.

        sym_x = sympy.Symbol("x")
        x = np.linspace(-2 * np.pi, 2 * np.pi, 100)
    
        def sin_expansion(x, n):
            return sympy.lambdify(sym_x, sympy.sin(sym_x).series(n=n+1).removeO(), 'numpy')(x)
    
        fig, ax = plt.subplots()
        ax.plot(x, np.sin(x), linewidth=4, color="red", label='exact')
        colors = ["blue", "black"]
        linestyles = [':', '-.', '--']
    
        for idx, n in enumerate(range(1, 12, 2)):
            ax.plot(x, sin_expansion(x, n), color=colors[idx // 3],
                linestyle=linestyles[idx % 3], linewidth=3,
                label="order %d approx." % (n+1))
    
        ax.set_ylim(-1.1, 1.1)
        ax.set_xlim(-1.5*np.pi, 1.5*np.pi)
    
        ax.legend(bbox_to_anchor=(1.02, 1), loc=2, borderaxespad=0.0)
        fig.subplots_adjust(right=.75)
    
    I highly recommend the book. It's full of nuggets like this.
  • kccqzy 7 years ago

    LaTeX doesn't have enough information about what your notations mean. You can very well write nonsensical formulas that look pretty in LaTeX but are absolutely meaningless.

    • bingerman 7 years ago

      I wish I had read the texbook or something similar sooner to gain knowledge like this. Used latex for years without knowing the basics and I regret that a lot.

      Also, (v)phantom and smash are something I really should have learned before all those fancy packages, nowadays I'm mostly using context anyways.

joseph8th 7 years ago

Any opinion on the utility of Emacs Org-mode to organize and manage LaTeX? In particular Org Babel?

  • p10_user 7 years ago

    I’ve written documents in org mode and converted to pdf via LaTeX, but I find that if the document gets sufficiently complicated with formatting, I have so many LaTeX blocks in my org file I might as well be writing LaTeX directly.

    Maybe I’m doing something wrong. YMMV

    • loskutak 7 years ago

      That is true, but org-mode really shines when you want to do literate programming stuff, e.g. have the matplotlib code directly in the orgfile, ...

tapia 7 years ago

I already implement most of the points mentioned there. The most useful (and new) tip for me was however the rasterization part. I normally like to have pdf figures for my LaTeX papers, but last time I had some graphics with some thousands of points plotted, which were taking too long to be printed if you did that from windows (in Linux there was no problem, that's why I didn't catch the problem earlier). At the end I decided to save the plot as png, but was not happy about it haha. It would have been good to know the rasterization trick earlier.

  • WookaiOP 7 years ago

    Indeed, it's pretty useful to be able to rasterize only parts of the plot! Glad you find it useful!

WookaiOP 7 years ago

Thanks all for the great feedback and discussion, I'll update this thread once I push an update. If you're interested, there was a great discussion on /r/MachineLearing as well: https://www.reddit.com/r/MachineLearning/comments/b2oiaj/d_b...

stilley2 7 years ago

Thanks for the write-up! Two notes from my experience: pgf output works well with latex as well (although will slowdown compilation), and I recommend not using the pyplot submodule, especially if you'll be running things remotely over ssh and don't have a display

billfruit 7 years ago

Is there a better and more comprehensive plotting library than Matplotlib, it's 3D plots a lack polish. Also it is kind of verbose and require much boilerplate. Its api is sprawling and hard to remember.

musicale 7 years ago

I have one tip for anyone using LaTeX:

Please stop using the awful Computer Modern typeface.

Keyboard Shortcuts

j
Next item
k
Previous item
o / Enter
Open selected item
?
Show this help
Esc
Close modal / clear selection