commit 76012f919f76f4326e20f439c470767cc819eaf8 Author: Cal Wing <20716204+calw20@users.noreply.github.com> Date: Mon Sep 30 14:48:16 2024 +1000 Add initial python files diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..820fdd4 --- /dev/null +++ b/.gitignore @@ -0,0 +1,171 @@ +# ---> Python +# Byte-compiled / optimized / DLL files +__pycache__/ +*.py[cod] +*$py.class + +# C extensions +*.so + +# Distribution / packaging +.Python +build/ +develop-eggs/ +dist/ +downloads/ +eggs/ +.eggs/ +lib/ +lib64/ +parts/ +sdist/ +var/ +wheels/ +share/python-wheels/ +*.egg-info/ +.installed.cfg +*.egg +MANIFEST + +# PyInstaller +# Usually these files are written by a python script from a template +# before PyInstaller builds the exe, so as to inject date/other infos into it. +*.manifest +*.spec + +# Installer logs +pip-log.txt +pip-delete-this-directory.txt + +# Unit test / coverage reports +htmlcov/ +.tox/ +.nox/ +.coverage +.coverage.* +.cache +nosetests.xml +coverage.xml +*.cover +*.py,cover +.hypothesis/ +.pytest_cache/ +cover/ + +# Translations +*.mo +*.pot + +# Django stuff: +*.log +local_settings.py +db.sqlite3 +db.sqlite3-journal + +# Flask stuff: +instance/ +.webassets-cache + +# Scrapy stuff: +.scrapy + +# Sphinx documentation +docs/_build/ + +# PyBuilder +.pybuilder/ +target/ + +# Jupyter Notebook +.ipynb_checkpoints + +# IPython +profile_default/ +ipython_config.py + +# pyenv +# For a library or package, you might want to ignore these files since the code is +# intended to run in multiple environments; otherwise, check them in: +# .python-version + +# pipenv +# According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control. +# However, in case of collaboration, if having platform-specific dependencies or dependencies +# having no cross-platform support, pipenv may install dependencies that don't work, or not +# install all needed dependencies. +#Pipfile.lock + +# poetry +# Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control. +# This is especially recommended for binary packages to ensure reproducibility, and is more +# commonly ignored for libraries. +# https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control +#poetry.lock + +# pdm +# Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control. +#pdm.lock +# pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it +# in version control. +# https://pdm.fming.dev/#use-with-ide +.pdm.toml + +# PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm +__pypackages__/ + +# Celery stuff +celerybeat-schedule +celerybeat.pid + +# SageMath parsed files +*.sage.py + +# Environments +.env +.venv +env/ +venv/ +ENV/ +env.bak/ +venv.bak/ + +# Spyder project settings +.spyderproject +.spyproject + +# Rope project settings +.ropeproject + +# mkdocs documentation +/site + +# mypy +.mypy_cache/ +.dmypy.json +dmypy.json + +# Pyre type checker +.pyre/ + +# pytype static type analyzer +.pytype/ + +# Cython debug symbols +cython_debug/ + +# PyCharm +# JetBrains specific template is maintained in a separate JetBrains.gitignore that can +# be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore +# and can be added to the global gitignore or merged into this file. For a more nuclear +# option (not recommended) you can uncomment the following to ignore the entire idea folder. +#.idea/ + +# Image Folder +images +images-* + +# Temp Folder +tmp/ +tmp-* + +data/~$* diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..770ce01 --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,24 @@ +{ + // Use IntelliSense to learn about possible attributes. + // Hover to view descriptions of existing attributes. + // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 + "version": "0.2.0", + "configurations": [ + { + "name": "Python: Main File", + "type": "python", + "request": "launch", + "program": "main.py", + "console": "integratedTerminal", + "justMyCode": true + }, + { + "name": "Python: Current File", + "type": "python", + "request": "launch", + "program": "${file}", + "console": "integratedTerminal", + "justMyCode": true + } + ] +} \ No newline at end of file diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..0e0dcd2 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,3 @@ +{ + +} \ No newline at end of file diff --git a/makeGraph.py b/makeGraph.py new file mode 100644 index 0000000..2d45619 --- /dev/null +++ b/makeGraph.py @@ -0,0 +1,540 @@ +### MatPlotLib Graph Wrapper +#### Written by Cal.W 2020, originally for MECH2700 but continually +#### expanded upon. +#### 2023 - Added UQ Colors +#### 2023 - Added pltKeyClose function +#### 2023 - Added UQ Default Colours to MatPlotLib +#### 2024 - Added Annotation & Fill + +__author__ = "Cal Wing" +__version__ = "0.1.11" + +from collections.abc import Iterator +import numpy as np +import matplotlib +import matplotlib.pyplot as plt +import matplotlib.colors as colors +from mpl_toolkits.axes_grid1 import make_axes_locatable +from cycler import cycler +import colorsys + +# Define the UQ Colours +UQ_COLOURS_DICT = { + # Primary + "purple": "#51247A", + "white" : "#FFFFFF", + "black" : "#000000", + + # Secondary + "light_purple": "#962A8B", + "red" : "#E62645", + "green" : "#2EA836", + "gold" : "#BB9D65", + "neutral" : "#D7D1CC", + "orange" : "#EB602B", + "yellow" : "#FBB800", + "blue" : "#4085C6", + "aqua" : "#00A2C7", + "dark_grey" : "#999490" +} + + +# Define a colour object that can do neat conversions & things, by default stores as hex value +class ColourValue(str): + def __new__(self, name, value): + self.name = name + self._orig_value = value + self.value = colors.to_hex(self._orig_value, True) + + return super().__new__(self, self.value) + + def __str__(self) -> str: + return self.value + + def __repr__(self) -> str: + return self.name + " " + self.value + " " + str(self.rgba()) + + def rgba(self, alpha = None) -> tuple[float, float, float, float]: + return colors.to_rgba(self.value, alpha) + + def rgb(self) -> tuple[float, float, float]: + return colors.to_rgb(self._orig_value) + + def hex(self) -> str: + return self.value + + def hsv(self) -> np.ndarray: + return colors.rgb_to_hsv(self.rgb()) + + def hls(self) -> tuple[float, float, float]: + return colorsys.rgb_to_hls(*self.rgb()) + + def lighten(self, amount=0.5) -> tuple[float, float, float]: + hls = self.hls() + + return colorsys.hls_to_rgb(hls[0], max(0, min(1, amount * hls[1])), hls[2]) + + def hex_lighten(self, amount=0.5) -> str: + return colors.to_hex(self.lighten(amount), True) + + + +# Define the UQ Colours in a nicer object +class ColourList(object): + def __init__(self, colours: dict) -> None: + self.colours = {} + + for key, value in colours.items(): + self.colours[key] = ColourValue(key, value) + + setattr(self, key, self.colours[key]) + + def __getitem__(self, key: str) -> str: + if key.replace(" ", "_") in self.colours.keys(): + key = key.replace(" ", "_") + + return self.colours[key] + + def items(self): + return self.colours.items() + + def __repr__(self) -> str: + return f"Colour List of {len(self.colours)} colour{'s' if len(self.colours) > 0 else ''}: " + str(list(self.colours.keys())) + +UQ_COLOURS = ColourList(UQ_COLOURS_DICT) + +# Load UQ Colours into MatPlotLib +# UQ colours are prefaced with 'uq:', so UQ red is 'uq:red' +# Note: Any names That have a _ also have a version with spaces so both "uq:light_purple" and "uq:light purple" work +uq_colour_mapping = {'uq:' + name: value for name, value in list(UQ_COLOURS.items()) + [(x[0].replace("_", " "), x[1]) for x in UQ_COLOURS.items() if "_" in x[0]]} +colors.get_named_colors_mapping().update( uq_colour_mapping ) + +## UQ Colour Cycler +# +-----------------------------+-----------------------------+ +# | Default (Tab) | UQ | +# +-----------------------------+-----------------------------+ +# | C00 | #1f77b4 -> tab:blue | #51247A -> uq:purple | +# | C01 | #ff7f0e -> tab:orange | #4085C6 -> uq:blue | +# | C02 | #2ca02c -> tab:green | #2EA836 -> uq:green | +# | C03 | #d62728 -> tab:red | #E62645 -> uq:red | +# | C04 | #9467bd -> tab:purple | #962A8B -> uq:light_purple | +# | C05 | #8c564b -> tab:brown | #999490 -> uq:dark_grey | +# | C06 | #e377c2 -> tab:pink | #EB602B -> uq:orange | +# | C07 | #7f7f7f -> tab:grey | #FBB800 -> uq:yellow | +# | C08 | #bcbd22 -> tab:olive | #00A2C7 -> uq:aqua | +# | C09 | #17becf -> tab:cyan | #BB9D65 -> uq:gold | +# | C10 | | #D7D1CC -> uq:neutral | +# +-----------------------------+-----------------------------+ + +# Build a colour cycler +uq_colour_cycler_factory = lambda: cycler(color=[ + UQ_COLOURS["purple"], #51247A -> C00 -> uq:purple + UQ_COLOURS["blue"], #4085C6 -> C01 -> uq:blue + UQ_COLOURS["green"], #2EA836 -> C02 -> uq:green + UQ_COLOURS["red"], #E62645 -> C03 -> uq:red + UQ_COLOURS["light_purple"], #962A8B -> C04 -> uq:light_purple + UQ_COLOURS["dark_grey"], #999490 -> C05 -> uq:dark_grey + UQ_COLOURS["orange"], #EB602B -> C06 -> uq:orange + UQ_COLOURS["yellow"], #FBB800 -> C07 -> uq:yellow + UQ_COLOURS["aqua"], #00A2C7 -> C08 -> uq:aqua + UQ_COLOURS["gold"], #BB9D65 -> C09 -> uq:gold + UQ_COLOURS["neutral"] #D7D1CC -> C10 -> uq:neutral +]) + +# Tell MatPlotLib to use said cycler +plt.rc('axes', prop_cycle=uq_colour_cycler_factory()) + + +## UQ Colour Gradient (Not very good :( ) +uq_colour_map_grad = [ + UQ_COLOURS["purple"], + UQ_COLOURS["light_purple"], + UQ_COLOURS["light_purple"], + UQ_COLOURS["blue"], + UQ_COLOURS["blue"], + UQ_COLOURS["aqua"], + UQ_COLOURS["green"], + UQ_COLOURS["green"], + UQ_COLOURS["green"], + UQ_COLOURS["yellow"], + UQ_COLOURS["yellow"] +] + +#Convert to RGB values +uq_colour_map_grad = [colors.to_rgb(c) for c in uq_colour_map_grad] + +#Populate the working dict +uq_colour_dict = { + "red": [], + "green": [], + "blue": [], +} +for i, c in enumerate(uq_colour_map_grad): + offset = i / (len(uq_colour_map_grad) - 1) + uq_colour_dict["red"].append( (offset, c[0], c[0]) ) + uq_colour_dict["green"].append( (offset, c[1], c[1]) ) + uq_colour_dict["blue"].append( (offset, c[2], c[2]) ) + +#Define & register the colour map itself +uq_cmap = colors.LinearSegmentedColormap('uq',segmentdata=uq_colour_dict) +matplotlib.colormaps.register(uq_cmap) + +# Set the colour map - Not a very good default so not doing that +#plt.set_cmap("uq") + + +## Colorbar Function by Joseph Long & Mike Lampton +# Retrieved from https://joseph-long.com/writing/colorbars/ on 31/10/2021 +# Minor Modifications made by Cal.W 2021 +def colorbar(mappable, size="5%", pad=0.05, lsize=None, lpad=None, lax=True, **kwargs): + last_axes = plt.gca() + ax = mappable.axes + fig = ax.figure + divider = make_axes_locatable(ax) + if lax: + lsize = lsize if lsize is not None else size + lpad = lpad if lpad is not None else pad + dax = divider.append_axes("left", size=lsize, pad=lpad) + dax.set_frame_on(False) + dax.grid(False) + dax.set_yticks([]) + dax.set_xticks([]) + cax = divider.append_axes("right", size=size, pad=pad) + cbar = fig.colorbar(mappable, cax=cax, **kwargs) + plt.sca(last_axes) + return cbar + + +## Make Graph Function +def makeGraph(graphData, showPlot=True, doProgramBlock=True, figSavePath=None, hideEmptyAxis=False, closeFig=False) -> tuple[matplotlib.figure.Figure, tuple[matplotlib.axes.Axes, ...]]: + """ Generate a matplotlib graph based on a simple dictionary object + Input: + dict(graphData): The dictionary containing all the graph data - see example for more info + bool(showPlot[True]): Should the function display the plot + bool(doProgramBlock[True]): Should the function block the main python thread + str(figSavePath[None]): The path to save a copy of the figure, calls fig.savefig if not None + + Returns: The the figure and axes from matplotlib.pyplot.subplots() + From 'matplotlib.pyplot.subplots(): + fig : `matplotlib.figure.Figure` + ax : `matplotlib.axes.Axes` or array of Axes + *ax* can be either a single `~matplotlib.axes.Axes` object or an + array of Axes objects if more than one subplot was created. + Example: + makeGraph({ + "title": "Simple Plot", + "xLabel": "x label", + "yLabel": "y label", + "plots": [ + {"x":[0,1,2,3,4], "y":[0,1,2,3,4], "label":"Linear"}, + {"x":[0,1,2,3,4], "y":[5,5,5,5,5]}, + {"x":[4,3,2,1,0], "y":[4,3,2,1,0], "label":"Linear2"}, + {"x":0, "type":"axvLine", "label":"Red Vertical Line", "color":"red"}, + {"y":6, "type":"axhLine", "label":"Dashed Horizontal Line", "args":{"linestyle":"--"}}, + {"type":"scatter", "x":4, "y":4, "label":"A Random Point", "colour":"purple", "args":{"zorder":2}} + ] + }) + """ + + doKeyCopy = True + plotDim = (1,) + if "subPlots" in graphData: + if "plotDim" in graphData: plotDim = graphData["plotDim"] + else: plotDim = (1,len(graphData["subPlots"])) + else: + graphData["subPlots"] = [graphData] + doKeyCopy = False + + figSize = graphData["figSize"] if "figSize" in graphData else None + + fig, axes = plt.subplots(*plotDim, figsize=figSize) # Create a figure and an axes. + + #if len(graphData["subPlots"]) <= 1: + # axes = [axes] + #Makes everything nice and linear + # IE ((1,2), (3,4)) = (1,2,3,4) + flatAxes = np.array(axes).flatten().tolist() + + loopKeys = [ + "xLabel", "yLabel", "title", "axis", "grid", "xPos", "yPos", + "xLabelPos", "yLabelPos", "xTickPos", "yTickPos", "xScale", "yScale", + "xTickMap", "yTickMap", "plots", "xLim", "yLim", "ledgLoc", "y2Label", + "ticklabel" + ] + + #Feel like this could be optimized + if doKeyCopy: + for key in loopKeys: + if key not in graphData: continue + if key in graphData: + for axGraphData in graphData["subPlots"]: + if key not in axGraphData: + axGraphData[key] = graphData[key] + + + for i, axGraphData in enumerate(graphData["subPlots"]): + ax1 = flatAxes[i] + + if bool(sum([("y2" in pData) for pData in axGraphData["plots"]])): + ax2 = ax1.twinx() + else: + ax2 = None + + # Duct Tape + ax = ax1 + + #Draw many plots as needed + # Also provide functions for drawing other types of lines + if "plots" in axGraphData: + for pData in axGraphData["plots"]: + getSafeValue = lambda key, result=None: pData[key] if key in pData else result #Only return the key-value if present in pData + getSafeValue2 = lambda key, key2, result=None: pData[key][key2] if key in pData and key2 in pData[key] else result + getSafeColour = getSafeValue("colour") or getSafeValue("color") #Frigen American Spelling + optArgs = getSafeValue("args", {}) #Allow for other args to be passed in + + if "x" in pData: + xData = pData["x"] + + if "y" in pData: + yData = pData["y"] + elif "y2" in pData: + yData = pData["y2"] + ax = ax2 + + if "type" not in pData or pData["type"] == "plot": + ax.plot(xData, yData, label=getSafeValue("label"), color=getSafeColour, **optArgs) + elif pData["type"] == "point": + ax.scatter(xData, yData, + marker=getSafeValue("marker"), label=getSafeValue("label"), + color=getSafeColour, zorder=getSafeValue("zorder", 2), + **optArgs ) + elif pData["type"] == "hLine": + ax.hlines(yData, *xData, label=getSafeValue("label"), color=getSafeColour, **optArgs) + elif pData["type"] == "vLine": + ax.vlines(xData, *yData, label=getSafeValue("label"), color=getSafeColour, **optArgs) + elif pData["type"] == "axvLine": + if "y" not in pData: yData = (0, 1) #Span the whole graph + ax.axvline(xData, *yData, label=getSafeValue("label"), color=getSafeColour, **optArgs) + elif pData["type"] == "axhLine": + if "x" not in pData: xData = (0, 1) #Span the whole graph + ax.axhline(yData, *xData, label=getSafeValue("label"), color=getSafeColour, **optArgs) + elif pData["type"] == "scatter": + ax.scatter(xData, yData, marker=getSafeValue("marker"), label=getSafeValue("label"), color=getSafeColour, **optArgs) + + elif pData["type"] == "contour": + cs = ax.contour(getSafeValue("x"), getSafeValue("y"), pData["z"], levels=getSafeValue("levels"), colors=getSafeColour, **optArgs) + if "label" in pData: cs.collections[0].set_label(getSafeValue("label")) + elif pData["type"] == "matshow": + ms = ax.matshow(pData["matrix"], origin=getSafeValue("origin"), label=getSafeValue("label"), **optArgs) + if "colourBar" in pData: + colorbar(ms, extend=getSafeValue2("colourBar", "extend")) + elif pData["type"] == "pColourMesh": + mesh = [] + if "X" in pData or "Y" in pData: + mesh = [xData, yData, pData["Z"]] + if "x" in pData or "y" in pData: + x = xData; y = yData + + if type(x) in [int, float]: x = (0, x, None) + if type(y) in [int, float]: y = (0, x, None) + x = tuple(x); y = tuple(y) + if len(x) < 3: x = (x[0], x[1], None) + if len(y) < 3: y = (y[0], y[1], None) + + x = np.arange(x[0], x[1], x[2]) + y = np.arange(y[0], y[1], y[2]) + X, Y = np.meshgrid(x, y) + + mesh = [X, Y, pData["Z"]] + else: + mesh = [pData["Z"]] + + cNorm = None + if "norm" in pData: + cNorm = colors.LogNorm(vmin=pData["norm"][0], vmax=pData["norm"][1]) + + pcMesh = ax.pcolormesh(*mesh, norm=cNorm, shading=getSafeValue("shading"), label=getSafeValue("label"), **optArgs) + #pcMesh = ax.imshow(pData["Z"], norm=cNorm, origin="lower") + + if "colourBar" in pData: + cBarOptArgs = pData["colourBar"]["optArgs"] if "optArgs" in pData["colourBar"] else {} + fig.colorbar(pcMesh, ax=ax, extend=getSafeValue2("colourBar", "extend"), **cBarOptArgs) + elif pData["type"] == "imshow": + cNorm = None + if "norm" in pData: + cNorm = colors.LogNorm(vmin=pData["norm"][0], vmax=pData["norm"][1]) + + ims = ax.imshow(pData["data"], norm=cNorm, origin=getSafeValue("origin"), label=getSafeValue("label"), **optArgs) + + if "colourBar" in pData: + cBarOptArgs = pData["colourBar"]["optArgs"] if "optArgs" in pData["colourBar"] else {} + colorbar(ims, extend=getSafeValue2("colourBar", "extend"), **cBarOptArgs) + elif pData["type"] == "text": + if not "props" in pData: + props = { + "boxstyle" : getSafeValue("boxstyle", "round"), + "facecolor": getSafeValue("facecolor", getSafeValue("facecolour", "wheat")), + "alpha" : getSafeValue("alpha", 0.5) + } + + align = ( + getSafeValue("valign", None), + getSafeValue("halign", None), + ) + align = getSafeValue("align", align) + + ax.text(getSafeValue("x", 0.05), getSafeValue("y", 0.95), pData["text"], transform=ax.transAxes, fontsize=getSafeValue("fontsize", None), va=align[0], ha=align[1], bbox=props) + elif pData["type"] == "annotate": + if type(pData["label"]) == str: + if "pos" in pData: + pData["x"] = pData["pos"][0] + pData["y"] = pData["pos"][1] + ax.annotate( + pData["label"], # this is the text + (pData["x"],pData["y"]), # these are the coordinates to position the label + textcoords=getSafeValue("offType", "offset points"), # how to position the text + xytext=getSafeValue("offset", (0,10)), # distance from text to points (x,y) + ha=getSafeValue("align", 'center') # horizontal alignment can be left, right or center + ) + else: + if "pos" not in pData: + pData["pos"] = list(zip(pData['x'], pData['y'])) + for i, label in enumerate(pData["label"]): + ax.annotate( + label, # this is the text + (pData["pos"][i][0], pData["pos"][i][1]), # these are the coordinates to position the label + textcoords=getSafeValue("offType", "offset points"), # how to position the text + xytext=getSafeValue("offset", (0,10)), # distance from text to points (x,y) + ha=getSafeValue("align", 'center') # horizontal alignment can be left, right or center + ) + elif pData["type"] == "fill": + ax.fill_between(list(pData["x"]), list(pData["y"]), color=getSafeColour, alpha=getSafeValue("alpha", 1), linewidth=getSafeValue("linewidth", None)) + #ax.fill_between(xA, yA, color="w") + + + + #Set extra options as needed + ax = ax1 + if "xLabel" in axGraphData: ax.set_xlabel(axGraphData["xLabel"]) # Add an x-label to the axes. + if "yLabel" in axGraphData: ax.set_ylabel(axGraphData["yLabel"]) # Add an y-label to the axes. + if "y2Label" in axGraphData: ax2.set_ylabel(axGraphData["y2Label"]) # Add a y2-label to the axes. + if "title" in axGraphData: ax.set_title(axGraphData["title"]) # Add an title to the axes. + if "axis" in axGraphData: ax.axis(axGraphData["axis"]) # Set the axis type + if "grid" in axGraphData: ax.grid(axGraphData["grid"]) # Add grids to the graph + if "xPos" in axGraphData: # Add the abilty to move the x axis label and ticks + ax.xaxis.set_label_position(axGraphData["xPos"]) + ax.xaxis.set_ticks_position(axGraphData["xPos"]) + if "yPos" in axGraphData: # Add the abilty to move the y axis label and ticks + ax.yaxis.set_label_position(axGraphData["yPos"]) + ax.yaxis.set_ticks_position(axGraphData["yPos"]) + if "xLabelPos" in axGraphData: ax.xaxis.set_label_position(axGraphData["xLabelPos"]) # Add the ability to move the x axis label + if "yLabelPos" in axGraphData: ax.yaxis.set_label_position(axGraphData["yLabelPos"]) # Add the ability to move the y axis label + if "xTickPos" in axGraphData: ax.xaxis.set_ticks_position(axGraphData["xTickPos"]) # Add the ability to move the x axis ticks + if "yTickPos" in axGraphData: ax.yaxis.set_ticks_position(axGraphData["yTickPos"]) # Add the ability to move the y axis ticks + if "xScale" in axGraphData: ax.set_xscale(axGraphData["xScale"]) #Add x axis scaling if needed + if "yScale" in axGraphData: ax.set_yscale(axGraphData["yScale"]) #Add y axis scaling if needed + if "xLim" in axGraphData: + xLimit = () + if type(axGraphData["xLim"]) in [int, float]: + xLimit = (0, axGraphData["xLim"]) + else: + xLimit = axGraphData["xLim"] + ax.set_xlim(xLimit) + if "yLim" in axGraphData: + yLimit = () + if type(axGraphData["yLim"]) in [int, float]: + yLimit = (0, axGraphData["yLim"]) + else: + yLimit = axGraphData["yLim"] + ax.set_ylim(yLimit) + if "xTickMap" in axGraphData: #Allow for the mapping / transformation of the xAxis Ticks + xTicks = matplotlib.ticker.FuncFormatter(lambda x, pos: '{0:g}'.format(axGraphData["xTickMap"](x))) + ax.xaxis.set_major_formatter(xTicks) + if "yTickMap" in axGraphData: #Allow for the mapping / transformation of the yAxis Ticks + yTicks = matplotlib.ticker.FuncFormatter(lambda y, pos: '{0:g}'.format(axGraphData["yTickMap"](y))) + ax.yaxis.set_major_formatter(yTicks) + + if "plots" in axGraphData and bool(sum([("label" in pData) for pData in axGraphData["plots"]])): + locPoint = axGraphData["ledgLoc"] if "ledgLoc" in axGraphData else None + lines1, labels1 = ax1.get_legend_handles_labels() + + if ax2: + lines2, labels2 = ax2.get_legend_handles_labels() + ax2.legend(lines1 + lines2, labels1 + labels2, loc=locPoint) + else: + ax1.legend(lines1, labels1, loc=locPoint) + + if "ticklabel" in axGraphData: + style = axGraphData["ticklabel"]["style"] if "style" in axGraphData["ticklabel"] else "" + axis = axGraphData["ticklabel"]["axis"] if "axis" in axGraphData["ticklabel"] else "both" + limits = axGraphData["ticklabel"]["limits"] if "limits" in axGraphData["ticklabel"] else None + optArgs = axGraphData["ticklabel"]["optArgs"] if "optArgs" in axGraphData["ticklabel"] else {} + ax.ticklabel_format(axis=axis, style=style, scilimits=limits, **optArgs) + + #Should work? + if hideEmptyAxis: + if not ax.collections and not ax.lines: + ax.set_axis_off() + + + if "title" in graphData and not "figTitle" in graphData: fig.canvas.manager.set_window_title(graphData["title"].replace("\n", " ")) #Set the figure title correctly + if "figTitle" in graphData: + getSafeValue = lambda key: graphData[key] if key in graphData else None #Only return the key-value if present in graphData + + fig.suptitle(graphData["figTitle"], fontsize=getSafeValue("figTitleFontSize")) + fig.canvas.manager.set_window_title(graphData["figTitle"].replace("\n", " ")) + if "windowTitle" in graphData: + fig.canvas.manager.set_window_title(graphData["windowTitle"].replace("\n", " ")) + + fig.tight_layout() #Fix labels being cut off sometimes + + #Very big hack + if hideEmptyAxis: + flatAxes[-1].set_axis_off() + + if figSavePath: + fig.savefig(figSavePath.format(fig.canvas.manager.get_window_title())) + + if showPlot: + plt.show(block=doProgramBlock) #Show the plot and also block the program - doing things OO style allow for more flexible programs + + if closeFig: + matplotlib.pyplot.close(fig) + + return fig, axes + +# [TODO] Make this Async so the closure of all graphs exits +def pltKeyClose(): + '''Show all plots and wait for user input to close them all.''' + plt.show(block=False) + input('Press any key to close all graphs...') + plt.close() + +if __name__ == '__main__': + #This is an example of drawing 4 plots by generating them + graphData = { + "figTitle": "Simple Plot", + "figTitleFontSize": 16, + "figSize": (8,8), #Yay America, this is in inches :/ # Note: cm = 1/2.54 + "xLabel": "x label", + "yLabel": "y label", + "plotDim": (2,2), + "subPlots":[] + } + + #Create 4 identical plots with different names + for i in range(4): + newPlot = { + "title": f"Graph {i+1}", + "plots": [ + {"x":[0,1,2,3,4], "y":[0,1,2,3,4], "label":"Linear"}, + {"x":[0,1,2,3,4], "y":[5,5,5,5,5]}, + {"x":[4,3,2,1,0], "y":[4,3,2,1,0], "label":"Linear2"}, + {"x":0, "type":"axvLine", "label":"Red Vertical Line", "color":"uq:red"}, + {"y":6, "type":"axhLine", "label":"Dashed Horizontal Line", "args":{"linestyle":"--"}}, + {"type":"point", "x":4, "y":4, "label":"A Random Point", "colour":"uq:purple"} + ] + } + graphData["subPlots"].append(newPlot) + + makeGraph(graphData) \ No newline at end of file diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 0000000..a4d073c Binary files /dev/null and b/requirements.txt differ