Empowering Controls Engineers with Codesys

Explore tips and examples for engineers transitioning to Codesys, enhancing their programming skills and efficiency in automation projects.

5+

10+

Pro Tips

Years of Codesys Experience

Benefits of OOP in Codesys

When it comes to PLC programming, efficiency, readability, and scalability are key factors in developing robust control systems. One of the most effective ways to achieve this is by leveraging Structured Text (ST) and Function Blocks (FBs)—powerful tools that not only make the code more modular but also enhance reusability and maintainability. If you are anything like me and came from primarily a Controls Engineering background, you are probably used to writing ladder logic code in a long and continuous procedural programming style.  This works, but can become very hard to maintain, and not very easily scalable.  Here in this post, I share a small introduction to an more Object Oriented Programming (OOP) approach to common challenges we face as Controls Engineers in the Industrial Controls and Automation field.

From Ladder Logic to Structured Text: A Cleaner Approach

Many PLC programmers start with Ladder Logic (LAD) due to its visual representation of electrical circuits. However, as control systems grow more complex, Structured Text (ST) emerges as a more concise and organized approach. It allows for logic to be written in a way that closely resembles high-level programming languages, making it easier to follow and troubleshoot.

Take, for example, a Motor Starter logic. Instead of implementing it in Ladder Logic with multiple contacts and coils, we can create a Function Block (FB) in Structured Text that encapsulates the entire logic.

IMPORTANT:  The code provided here is for reference only, if you are writing logic for a motor starter, or a motor drive, please follow specific controls narrative and specification that is provided by your mechanical engineering or application engineering team.  The code that I am providing here is just an example, and although it is inspired by my 14 years of experience in the ICS field, it is not code that has been tested on real equipment.  I do not take any responsibility for damages done to personnel or equipment if you decide to copy this code directly to your project and use it without it being thoroughly tested. 

Ok.. after that needed, disclaimer for safety reasons now lets jump into the fun part!

Below is an example of how this logic might look like in traditional LAD code.

This works and it is sometimes, depending on your level of exposure to LAD, easy to follow.  But it is error prone and very hard to expand upon.  If you make one small change in your motor drive / starter logic now you have to go and change it everywhere in your LAD code.  Which as you can imagine it is not efficient at all.

Encapsulating Logic with Function Blocks

A Function Block is essentially a reusable module that takes inputs, processes logic, and provides outputs. Here’s how we define our Motor Starter FB in Structured Text:

This block efficiently manages the motor starter logic, handling start conditions, overload protection, and feedback monitoring in a structured and reusable format.

Reusability and Scalability: The Game-Changer

With our MotorStarter function block in place, we can now instantiate multiple motor starters within a program, allowing for clean and scalable control implementations.

VAR
     motorStarter:      ARRAY[1..10] OF MotorStarter;
     motorP :                ARRAY[1..10] OF MTR_MotorStarter;
END_VAR

By simply declaring an array of Motor Starter objects, we can manage multiple motors without redundant logic. Each instance operates independently, making it easy to expand the system as needed.

Unit Testing with Function Blocks

One of the best advantages of this approach is that unit testing becomes straightforward. Using Continuous Function Chart (CFC) or other visualization tools, we can instantiate multiple instances (e.g., motorStarter[1], motorStarter[2]) and clearly see how inputs affect outputs in real time.

This structured approach makes debugging significantly easier, as each instance retains its own state and can be tested independently before integrating into a larger automation system.

Conclusion:
A Step Toward Modular Automation

By utilizing Structured Text and Function Blocks, PLC programmers can write cleaner, more maintainable, and scalable code. Whether managing a single motor starter or an entire production line, this approach ensures consistency and efficiency.

This is just the beginning! Future discussions will explore how these principles integrate into larger automation systems, unlocking even greater possibilities. Stay tuned for more insights!

CODESYS Pro Tips

Explore essential tips and tricks for engineers transitioning to CODESYS programming platform effectively.

Write Values [CTRL + F7]

Specially useful when in simulation mode and doing unit testing, if you need to write a value into a variable to test the functionality of your code. Prepare the value by typing it in CODESYS then hit CTR + F7 on your keyboard, this will write the values you prepared to the variables you are working with. Similarly you can do this by going to Debug -> Write Values. It's just faster to do it on the keyboard.

Multiple Watch Tabs

When testing your application is specially useful to setup watches, if you work with only one watch window this can quickly become long and messy. Instead go to View -> Watch to add up to 4 watch windows. Then drag and drop the variables you want to monitor or modify their values for testing or troubleshooting. You can have one watch for the input the other for the outputs etc.

Auto-fill [CTRL + Space]

When working with ST, lets say you have an action called AnalogInputTesting, just start by typing some of the letters or words of the action, variable, parameter etc. you want to reference and then hit CTRL+Space; most of the times CODESYS will auto-fill the rest for you. e.g. Type up to AnalogIn -> [CTRL + Space] should result in AnalogInputTesting. Given that the action with that name exists in your code.

Variable Quick Declaration [CTRL + . ]

When you are building/writing your application sometimes you realize you need a variable you forgot to declare. In this case, just write the variable name, for example iMyVarInt, CODESYS's "intellisense" will draw a double red line under it. Now bring the cursor over the variable name you just wrote, and hit CTRL + . this will give you a small window with the option to Declare variable on the fly!

Variable Quick Refactoring [CTRL + . ]

This works sometimes, and I have found it to not work well in some other instances. But when it does, it makes refactoring (renaming) variables a breeze. Lets say the iMyVarInt you created above is not the name you want for this variable. Instead of going to the variable declaration scope and right clicking and selecting to refactor, just change the name of the variable. CODESYS will recognize that, the variable is not declared under the new name, and draw a double red underline. Again hit CTRL + . you will get the window shown below. Now choose Rename 'iMyVarInt' to 'iMyVarInteger' hit OK on the window that appears and you are done!

Multi Cursor Selection [SHIFT + ALT + UP or Down Arrow Keys ]

This is super useful when programming in Structured Text. Lets take the example above, and now lets say your iMyVarInteger is an array with 10 elements. You copied over the variable a couple of lines and wish to add a [ ] to all of them. Simply hit SHIFT + ALT and use the up and down arrow key for multi-cursor selection and start typing away. Now this is great but next I will share with you my favorite way to modify lines of code at once using Notepad++

Multi Cursor Selection using Notepad++

Ok so CODESYS IDE is great but it has it's limitations, this is where we have to get creative with our tool choices. For some tasks like this one I prefer Notepad++. Lets take the example above, now we have an array of INT for which the variable is called iMyVarInteger (probably should be Integer(s) but lets leave it like that for the example). Copy the iMyVarInteger := 10; line to Notepad++. Now use CTRL + D to duplicate the line 10 times. Use SHIFT + ALT + Arrow Keys for multi-cursor selection to add the [ ]. Finally place your multi cursor selection in between the brackets as shown below. Then hit ALT + C. You will be Presented with Column/Multi-Selection Editor, select Number to Insert, set your initial number, and your increase by and hit OK. You have just created 10 declarations for each element of your array. Copy and paste it back to CODESYS IDE, and modify or adjust as needed. This to me is one of the greatest benefits of using Structured Text in CODESYS. You can now use in conjunctions tools like, Notepad++, VSCode, and even run Copilot in VSCode after installing the Structured Text extension.
Once you get used to having a work flow with different tools like this, you can copy and paste edit code from one text editor to the other, boosting your productivity!

Quick Sanity Check - Generate Code [F11]

Once you get working with CODESYS for a while, and the size of your application grows. You start noticing that CODESYS is sometimes not very good at catching syntax errors automatically. So hitting the F11 key from time to time is a very good pro-tip. To me this is similar to hitting CTRL + S to save your application often in-case you experience a crash so you won't lose all your work!

Productivity Booster - Cycle Through Tabs [CTRL + Tab]

This is a huge productivity booster for me. Once your application grows and you have many tabs opened, sometimes you have to reference code that is on a different tab. Using your mouse to open the tab that you are looking for is good but you can quickly achieve the same results with CTRL + Tab. Reference the code you needed to reference and hit CTRL + Tab and you are back again on the pane that you were working on!

Where are My Comments? - FDB, LD, and IL Editors

This one tripped me up for a long time when I first started working with CODESYS—where did my comments go in my ladder logic code? It turns out there's a setting for that! To enable comments, go to Tools → Options → FBD, LD, and IL Editor, then check the box for Show network comment. Problem solved!

Input Assistant - [F2]

This one is a game-changer in Structured Text! Let's say you have a function block or function that takes multiple inputs and generates multiple outputs. When you start typing, for example, myFunction(, CODESYS will display a pop-up suggesting the required inputs and outputs. However, you’d still have to type each one manually, which can be tedious. Instead, press F2 to open the Input Assistant. Start typing the name of the function or function block you want to use—it should appear in the search results. Select it, ensure Insert with arguments is checked, then hit OK. CODESYS will automatically insert the function or function block with all its arguments, saving you time! Now, all that's left is to assign the correct variables to each parameter.

Get In Touch

macbook pro on black textile
macbook pro on black textile

Contact us for tips and resources on transitioning to Codesys for controls engineering projects.

woman wearing yellow long-sleeved dress under white clouds and blue sky during daytime

Do you want more content like this? Please share with us your thoughts and experience with CODESYS. Let us know if you want specific tech-tips with CODESYS!