2019-05-28 21:16:50 +02:00
|
|
|
This document provides information on implementing G-Code command
|
|
|
|
sequences in gcode_macro (and similar) config sections.
|
|
|
|
|
2020-01-06 16:59:40 +01:00
|
|
|
### G-Code Macro Naming
|
2020-01-08 01:06:55 +01:00
|
|
|
|
|
|
|
Case is not important for the G-Code macro name - MY_MACRO and
|
2020-01-06 16:59:40 +01:00
|
|
|
my_macro will evaluate the same and may be called in either upper or
|
2020-01-08 02:58:39 +01:00
|
|
|
lower case. If any numbers are used in the macro name then they must
|
|
|
|
all be at the end of the name (eg, TEST_MACRO25 is valid, but
|
|
|
|
MACRO25_TEST3 is not).
|
2020-01-06 16:59:40 +01:00
|
|
|
|
2019-05-28 21:16:50 +02:00
|
|
|
### Formatting of G-Code in the config
|
|
|
|
|
|
|
|
Indentation is important when defining a macro in the config file. To
|
|
|
|
specify a multi-line G-Code sequence it is important for each line to
|
|
|
|
have proper indentation. For example:
|
|
|
|
|
|
|
|
```
|
|
|
|
[gcode_macro blink_led]
|
|
|
|
gcode:
|
|
|
|
SET_PIN PIN=my_led VALUE=1
|
|
|
|
G4 P2000
|
|
|
|
SET_PIN PIN=my_led VALUE=0
|
|
|
|
```
|
|
|
|
|
|
|
|
Note how the `gcode:` config option always starts at the beginning of
|
|
|
|
the line and subsequent lines in the G-Code macro never start at the
|
|
|
|
beginning.
|
|
|
|
|
|
|
|
### Save/Restore state for G-Code moves
|
|
|
|
|
|
|
|
Unfortunately, the G-Code command language can be challenging to use.
|
|
|
|
The standard mechanism to move the toolhead is via the `G1` command
|
|
|
|
(the `G0` command is an alias for `G1` and it can be used
|
|
|
|
interchangeably with it). However, this command relies on the "G-Code
|
|
|
|
parsing state" setup by `M82`, `M83`, `G90`, `G91`, `G92`, and
|
|
|
|
previous `G1` commands. When creating a G-Code macro it is a good
|
|
|
|
idea to always explicitly set the G-Code parsing state prior to
|
|
|
|
issuing a `G1` command. (Otherwise, there is a risk the `G1` command
|
|
|
|
will make an undesirable request.)
|
|
|
|
|
|
|
|
A common way to accomplish that is to wrap the `G1` moves in
|
|
|
|
`SAVE_GCODE_STATE`, `G91`, and `RESTORE_GCODE_STATE`. For example:
|
|
|
|
|
|
|
|
```
|
|
|
|
[gcode_macro MOVE_UP]
|
|
|
|
gcode:
|
|
|
|
SAVE_GCODE_STATE NAME=my_move_up_state
|
|
|
|
G91
|
|
|
|
G1 Z10 F300
|
|
|
|
RESTORE_GCODE_STATE NAME=my_move_up_state
|
|
|
|
```
|
|
|
|
|
|
|
|
The `G91` command places the G-Code parsing state into "relative move
|
|
|
|
mode" and the `RESTORE_GCODE_STATE` command restores the state to what
|
|
|
|
it was prior to entering the macro. Be sure to specify an explicit
|
|
|
|
speed (via the `F` parameter) on the first `G1` command.
|
|
|
|
|
|
|
|
### Template expansion
|
2019-06-04 16:31:03 +02:00
|
|
|
<!-- {% raw %} -->
|
2019-05-28 21:16:50 +02:00
|
|
|
|
|
|
|
The gcode_macro `gcode:` config section is evaluated using the Jinja2
|
|
|
|
template language. One can evaluate expressions at run-time by
|
|
|
|
wrapping them in `{ }` characters or use conditional statements
|
|
|
|
wrapped in `{% %}`. See the
|
|
|
|
[Jinja2 documentation](http://jinja.pocoo.org/docs/2.10/templates/)
|
|
|
|
for further information on the syntax.
|
|
|
|
|
|
|
|
This is most often used to inspect parameters passed to the macro when
|
|
|
|
it is called. These parameters are available via the `params`
|
|
|
|
pseudo-variable. For example, if the macro:
|
|
|
|
|
|
|
|
```
|
|
|
|
[gcode_macro SET_PERCENT]
|
|
|
|
gcode:
|
|
|
|
M117 Now at { params.VALUE|float * 100 }%
|
|
|
|
```
|
|
|
|
|
|
|
|
were invoked as `SET_PERCENT VALUE=.2` it would evaluate to `M117 Now
|
|
|
|
at 20%`. Note that parameter names are always in upper-case when
|
|
|
|
evaluated in the macro and are always passed as strings. If performing
|
|
|
|
math then they must be explicitly converted to integers or floats.
|
|
|
|
|
|
|
|
An example of a complex macro:
|
|
|
|
```
|
|
|
|
[gcode_macro clean_nozzle]
|
|
|
|
gcode:
|
|
|
|
SAVE_GCODE_STATE NAME=clean_nozzle_state
|
|
|
|
G90
|
|
|
|
G0 Z15 F300
|
|
|
|
{% for wipe in range(8) %}
|
|
|
|
{% for coordinate in [(275,4),(235,4)] %}
|
|
|
|
G0 X{coordinate[0]} Y{coordinate[1] + 0.25 * wipe} Z9.7 F12000
|
|
|
|
{% endfor %}
|
|
|
|
{% endfor %}
|
|
|
|
RESTORE_GCODE_STATE NAME=clean_nozzle_state
|
|
|
|
```
|
2019-06-04 16:31:03 +02:00
|
|
|
<!-- {% endraw %} -->
|
2019-05-28 21:16:50 +02:00
|
|
|
|
|
|
|
#### The "printer" Variable
|
|
|
|
|
|
|
|
It is possible to inspect (and alter) the current state of the printer
|
|
|
|
via the `printer` pseudo-variable. For example:
|
|
|
|
|
|
|
|
```
|
|
|
|
[gcode_macro slow_fan]
|
|
|
|
gcode:
|
|
|
|
M106 S{ printer.fan.speed * 0.9 * 255}
|
|
|
|
```
|
|
|
|
|
|
|
|
Important! Macros are first evaluated in entirety and only then are
|
|
|
|
the resulting commands executed. If a macro issues a command that
|
|
|
|
alters the state of the printer, the results of that state change will
|
|
|
|
not be visible during the evaluation of the macro. This can also
|
|
|
|
result in subtle behavior when a macro generates commands that call
|
|
|
|
other macros, as the called macro is evaluated when it is invoked
|
|
|
|
(which is after the entire evaluation of the calling macro).
|
|
|
|
|
|
|
|
By convention, the name immediately following `printer` is the name of
|
|
|
|
a config section. So, for example, `printer.fan` refers to the fan
|
|
|
|
object created by the `[fan]` config section. There are some
|
|
|
|
exceptions to this rule - notably the `gcode` and `toolhead` objects.
|
|
|
|
If the config section contains spaces in it, then one can access it
|
|
|
|
via the `[ ]` accessor - for example:
|
|
|
|
`printer["generic_heater my_chamber_heater"].temperature`.
|
|
|
|
|
|
|
|
Some printer objects allow one to alter the state of the printer. By
|
|
|
|
convention, these objects use an `action_` prefix. For example,
|
|
|
|
`printer.gcode.action_emergency_stop()` would cause the printer to go
|
|
|
|
into a shutdown state. These actions are taken at the time that the
|
|
|
|
macro is evaluated, which may be a significant amount of time before
|
|
|
|
the generated commands are executed.
|
|
|
|
|
|
|
|
The following are common printer attributes:
|
|
|
|
- `printer.fan.speed`: The fan speed as a float between 0.0 and 1.0.
|
|
|
|
- `printer.gcode.action_respond_info(msg)`: Write the given `msg` to
|
|
|
|
the /tmp/printer pseudo-terminal. Each line of `msg` will be sent
|
|
|
|
with a "// " prefix.
|
|
|
|
- `printer.gcode.action_respond_error(msg)`: Write the given `msg` to
|
|
|
|
the /tmp/printer pseudo-terminal. The first line of `msg` will be
|
|
|
|
sent with a "!! " prefix and subsequent lines will have a "// "
|
|
|
|
prefix.
|
|
|
|
- `printer.gcode.action_emergency_stop(msg)`: Transition the printer
|
|
|
|
to a shutdown state. The `msg` parameter is optional, it may be
|
|
|
|
useful to describe the reason for the shutdown.
|
|
|
|
- `printer.gcode.gcode_position`: The current position of the toolhead
|
|
|
|
relative to the current G-Code origin. It is possible to access the
|
|
|
|
x, y, z, and e components of this position (eg,
|
|
|
|
`printer.gcode.gcode_position.x`).
|
|
|
|
- `printer["gcode_macro <macro_name>"].<variable>`: The current value
|
|
|
|
of a gcode_macro variable.
|
|
|
|
- `printer.<heater>.temperature`: The last reported temperature (in
|
2019-11-25 01:02:49 +01:00
|
|
|
Celsius as a float) for the given heater. Example heaters are:
|
|
|
|
`extruder`, `extruder1`, `heater_bed`, `heater_generic
|
|
|
|
<config_name>`.
|
2019-05-28 21:16:50 +02:00
|
|
|
- `printer.<heater>.target`: The current target temperature (in
|
|
|
|
Celsius as a float) for the given heater.
|
|
|
|
- `printer.pause_resume.is_paused`: Returns true if a PAUSE command
|
|
|
|
has been executed without a corresponding RESUME.
|
|
|
|
- `printer.toolhead.position`: The last commanded position of the
|
|
|
|
toolhead relative to the coordinate system specified in the config
|
|
|
|
file. It is possible to access the x, y, z, and e components of this
|
|
|
|
position (eg, `printer.toolhead.position.x`).
|
2019-11-25 01:02:49 +01:00
|
|
|
- `printer.toolhead.extruder`: The name of the currently active
|
|
|
|
extruder. For example, one could use
|
|
|
|
`printer[printer.toolhead.extruder].target` to get the target
|
|
|
|
temperature of the current extruder.
|
2019-11-25 01:16:21 +01:00
|
|
|
- `printer.toolhead.homed_axes`: The current cartesian axes considered
|
|
|
|
to be in a "homed" state. This is a string containing one or more of
|
|
|
|
"x", "y", "z".
|
2020-04-26 23:30:57 +02:00
|
|
|
- `printer.heaters.available_heaters`: Returns a list of all currently
|
2019-12-12 13:33:23 +01:00
|
|
|
available heaters by their full config section names,
|
|
|
|
e.g. `["extruder", "heater_bed", "heater_generic my_custom_heater"]`.
|
2020-04-26 23:30:57 +02:00
|
|
|
- `printer.heaters.available_sensors`: Returns a list of all currently
|
2019-12-12 13:33:23 +01:00
|
|
|
available temperature sensors by their full config section names,
|
|
|
|
e.g. `["extruder", "heater_bed", "heater_generic my_custom_heater",
|
|
|
|
"temperature_sensor electronics_temp"]`.
|
2020-02-13 01:11:40 +01:00
|
|
|
- `printer.query_endstops.last_query["<endstop>"]`: Returns True if
|
|
|
|
the given endstop was reported as "triggered" during the last
|
|
|
|
QUERY_ENDSTOP command. Note, due to the order of template expansion
|
|
|
|
(see above), the QUERY_STATUS command must be run prior to the macro
|
|
|
|
containing this reference.
|
2020-02-14 03:57:41 +01:00
|
|
|
- `printer.configfile.config["<section>"]["<option>"]`: Returns the
|
|
|
|
given config file setting as read by Klipper during the last
|
|
|
|
software start or restart. (Any settings changed at run-time will
|
|
|
|
not be reflected here.) All values are returned as strings (if math
|
|
|
|
is to be performed on the value then it must be converted to a
|
|
|
|
Python number).
|
2019-05-28 21:16:50 +02:00
|
|
|
|
|
|
|
The above list is subject to change - if using an attribute be sure to
|
2019-06-04 16:39:43 +02:00
|
|
|
review the [Config Changes document](Config_Changes.md) when upgrading
|
|
|
|
the Klipper software. The above list is not exhaustive. Other
|
|
|
|
attributes may be available (via `get_status()` methods defined in the
|
2019-05-28 21:16:50 +02:00
|
|
|
software). However, undocumented attributes may change without notice
|
|
|
|
in future Klipper releases.
|
|
|
|
|
|
|
|
### Variables
|
|
|
|
|
|
|
|
The SET_GCODE_VARIABLE command may be useful for saving state between
|
2020-01-06 16:59:40 +01:00
|
|
|
macro calls. Variable names may not contain any upper case characters.
|
|
|
|
For example:
|
2019-05-28 21:16:50 +02:00
|
|
|
|
|
|
|
```
|
|
|
|
[gcode_macro start_probe]
|
|
|
|
variable_bed_temp: 0
|
|
|
|
gcode:
|
|
|
|
# Save target temperature to bed_temp variable
|
|
|
|
SET_GCODE_VARIABLE MACRO=start_probe VARIABLE=bed_temp VALUE={printer.heater_bed.target}
|
|
|
|
# Disable bed heater
|
|
|
|
M140
|
|
|
|
# Perform probe
|
|
|
|
PROBE
|
|
|
|
# Call finish_probe macro at completion of probe
|
|
|
|
finish_probe
|
|
|
|
|
|
|
|
[gcode_macro finish_probe]
|
|
|
|
gcode:
|
|
|
|
# Restore temperature
|
|
|
|
M140 S{printer["gcode_macro start_probe"].bed_temp}
|
|
|
|
```
|
|
|
|
|
|
|
|
Be sure to take the timing of macro evaluation and command execution
|
|
|
|
into account when using SET_GCODE_VARIABLE.
|
2019-06-29 23:13:35 +02:00
|
|
|
|
|
|
|
### Delayed Gcodes
|
|
|
|
|
|
|
|
The [delayed_gcode] configuration option can be used to execute a delayed
|
|
|
|
gcode sequence:
|
|
|
|
|
|
|
|
```
|
|
|
|
[delayed_gcode clear_display]
|
|
|
|
gcode:
|
|
|
|
M117
|
|
|
|
|
|
|
|
[gcode_macro load_filament]
|
|
|
|
gcode:
|
|
|
|
G91
|
|
|
|
G1 E50
|
|
|
|
G90
|
|
|
|
M400
|
|
|
|
M117 Load Complete!
|
|
|
|
UPDATE_DELAYED_GCODE ID=clear_display DURATION=10
|
|
|
|
```
|
|
|
|
|
|
|
|
When the `load_filament` macro above executes, it will display a
|
|
|
|
"Load Complete!" message after the extrusion is finished. The
|
|
|
|
last line of gcode enables the "clear_display" delayed_gcode, set
|
|
|
|
to execute in 10 seconds.
|
|
|
|
|
|
|
|
The `initial_duration` config option can be set to execute the
|
|
|
|
delayed_gcode on printer startup. The countdown begins when the
|
|
|
|
printer enters the "ready" state. For example, the below delayed_gcode
|
|
|
|
will execute 5 seconds after the printer is ready, initializing
|
|
|
|
the display with a "Welcome!" message:
|
|
|
|
|
|
|
|
```
|
|
|
|
[delayed_gcode welcome]
|
|
|
|
initial_duration: 5.
|
|
|
|
gcode:
|
|
|
|
M117 Welcome!
|
|
|
|
```
|
|
|
|
|
|
|
|
Its possible for a delayed gcode to repeat by updating itself in
|
|
|
|
the gcode option:
|
|
|
|
|
|
|
|
```
|
|
|
|
[delayed_gcode report_temp]
|
|
|
|
initial_duration: 2.
|
|
|
|
gcode:
|
|
|
|
{printer.gcode.action_respond_info(
|
|
|
|
"Extruder Temp: %.1f" %
|
|
|
|
(printer.extruder0.temperature))}
|
|
|
|
UPDATE_DELAYED_GCODE ID=report_temp DURATION=2
|
|
|
|
|
|
|
|
```
|
|
|
|
|
|
|
|
The above delayed_gcode will send "// Extruder Temp: [ex0_temp]" to
|
|
|
|
Octoprint every 2 seconds. This can be canceled with the following
|
|
|
|
gcode:
|
|
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
UPDATE_DELAYED_GCODE ID=report_temp DURATION=0
|
|
|
|
```
|