AdvancedHMI Software

General Category => Open Discussion => Topic started by: joko markono on September 04, 2019, 03:59:57 AM

Title: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 04, 2019, 03:59:57 AM
Hi Guys,

My PLC is connected to serial COM1, and i want to send out the displayed/collected data (as a single line data string) to serial COM2.
How can i do that in AHMI?
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 04, 2019, 03:18:21 PM
I would suggest you read this:

https://docs.microsoft.com/en-us/dotnet/visual-basic/developing-apps/programming/computer-resources/how-to-send-strings-to-serial-ports

and use Google for any/all examples in VB.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 05, 2019, 01:57:28 AM
ok, actually i imported some code from
https://github.com/bartashevich/serial-communication (https://github.com/bartashevich/serial-communication)
I'm not very sure if my way of combining the PLC data (as you can see from 'SendTextBox' is correct. I actually just plus both data like this:

SendTextBox.Text = BasicLabel1.Text + BasicLabel2.Text

But so far the program is running fine. Here is the code for my DataPanelForm:
Code: [Select]
Public Class DataPanel

    Dim Data_received As String

    '*******************************************************************************
    '* Stop polling when the form is not visible in order to reduce communications
    '* Copy this section of code to every new form created
    '*******************************************************************************
    Private NotFirstShow As Boolean

    Private Sub Form_VisibleChanged(ByVal sender As Object, ByVal e As System.EventArgs) Handles Me.VisibleChanged
        '* Do not start comms on first show in case it was set to disable in design mode
        If NotFirstShow Then
            AdvancedHMIDrivers.Utilities.StopComsOnHidden(components, Me)
        Else
            NotFirstShow = True
        End If
    End Sub

    Private Sub ReturnToMainButton_Click(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Button2.Click
        MainForm.Show()
        ' Me.Hide()
        Me.Visible = False
    End Sub

    Private Sub MessageListByItem1_SelectedIndexChanged(sender As Object, e As EventArgs)

    End Sub

    Private Sub MessageListByValue1_SelectedIndexChanged(sender As Object, e As EventArgs)

    End Sub

    Private Sub SendTextBox_TextChanged(sender As Object, e As EventArgs) Handles SendTextBox.TextChanged
        'combined data = data1 + data2 + data3 + .....
        SendTextBox.Text = BasicLabel1.Text + BasicLabel2.Text

    End Sub

    Private Sub DataPanel_Load(sender As Object, e As EventArgs) Handles MyBase.Load
        Timer1.Interval = 300
        Timer1.Enabled = False

        ModuleComPortParameters.configuration_is_valid = False
        SendBtn.Enabled = False
    End Sub

    Private Sub ConfigurePortBtn_Click(sender As Object, e As EventArgs) Handles ConfigurePortBtn.Click
        ConfigurationPort.StartPosition = FormStartPosition.CenterParent
        ConfigurationPort.ShowDialog()

        If ModuleComPortParameters.configuration_is_valid = True Then
            With SerialPort1
                .PortName = ModuleComPortParameters.Port
                .BaudRate = ModuleComPortParameters.Baudrate
                .Parity = ModuleComPortParameters.Parity
            End With
        End If
    End Sub

    Private Sub ExitBtn_Click(sender As Object, e As EventArgs) Handles ExitBtn.Click
        End
    End Sub

    Private Sub OpenPortBtn_Click(sender As Object, e As EventArgs) Handles OpenPortBtn.Click
        If ModuleComPortParameters.configuration_is_valid = True Then
            If SerialPort1.IsOpen Then
                SerialPort1.Close()
                SendBtn.Enabled = False
                OpenPortBtn.Text = "Open Port"
                OpenPortBtn.BackColor = Color.LightGray
                Timer1.Enabled = False
            Else
                SerialPort1.Open()
                SendBtn.Enabled = True
                OpenPortBtn.Text = "Close Port"
                OpenPortBtn.BackColor = Color.Red
                Timer1.Enabled = True
            End If
        Else
            MsgBox("Cannot open port with an invalid configuration")
        End If
    End Sub

    Private Sub SendBtn_Click(sender As Object, e As EventArgs) Handles SendBtn.Click
        SerialPort1.Write(SendTextBox.Text)
    End Sub

    Private Sub Timer1_Tick(sender As Object, e As EventArgs) Handles Timer1.Tick
        Data_received &= SerialPort1.ReadExisting
        If Len(Data_received) > 0 Then
            ReceiveTextBox.Text = Data_received
            HistoryTextBox.Text = TimeOfDay.ToLongTimeString + " : " + Data_received + vbNewLine + HistoryTextBox.Text
            Data_received = ""
        End If
    End Sub
End Class


and this the code for the ConfigurationPort:
Code: [Select]
Public Class ConfigurationPort
    Private Sub Port_Conf_Load(sender As Object, e As EventArgs) Handles MyBase.Load
        Dim available_Ports As Array = IO.Ports.SerialPort.GetPortNames
        Dim i As Integer

        For i = 0 To UBound(available_Ports)
            PortComboBox.Items.Add(available_Ports(i))
        Next

        BaudrateComboBox.Items.Clear()
        BaudrateComboBox.Items.Add("4800")
        BaudrateComboBox.Items.Add("9600")
        BaudrateComboBox.Items.Add("19200")
        BaudrateComboBox.SelectedIndex = 2

        ParityComboBox.Items.Clear()
        ParityComboBox.Items.Add(IO.Ports.Parity.None)
        ParityComboBox.Items.Add(IO.Ports.Parity.Odd)
        ParityComboBox.Items.Add(IO.Ports.Parity.Even)
        ParityComboBox.SelectedIndex = 2

    End Sub

    Private Sub Cancel_Btn_Click(sender As Object, e As EventArgs) Handles CancelBtn.Click
        ModuleComPortParameters.configuration_is_valid = False
        Me.Close()
    End Sub

    Private Sub Ok_Btn_Click(sender As Object, e As EventArgs) Handles OkBtn.Click
        ModuleComPortParameters.Port = PortComboBox.Text
        ModuleComPortParameters.baudrate = BaudrateComboBox.Text

        Select Case ParityComboBox.Text
            Case "None"
                ModuleComPortParameters.parity = IO.Ports.Parity.None
            Case "Even"
                ModuleComPortParameters.parity = IO.Ports.Parity.Even
            Case "Odd"
                ModuleComPortParameters.parity = IO.Ports.Parity.Odd

        End Select
        ModuleComPortParameters.configuration_is_valid = True

        Me.Close()
    End Sub

    Private Sub Port_combo_SelectedIndexChanged(sender As Object, e As EventArgs) Handles PortComboBox.SelectedIndexChanged

    End Sub
End Class

The results is shown in the attachment. I just need to know if the way i combined the data is correct. Currently i don't have my PLC data connected.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 05, 2019, 02:39:59 AM
It's usually "&" instead of "+", but if it works then good.

com0com program is free and for virtual ports is usually the best choice, you can find the link here:

https://www.advancedhmi.com/forum/index.php?topic=1961.0

I'm not sure if your Eltima software is free.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 05, 2019, 10:06:02 PM
i have problem to read to incoming data.
there is no display on the textbox.
I notice the data has a kind of weird format like these:

$PPTIROV (NMEA 0183 format)

is there a way i can assigned the format (any type) in the beginning of my receiving text box?
Some other data start with different format like "$NORCLOG", ":" , "$", etc....
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 01:45:42 AM
This sounds like something Archie would have to decipher.

Maybe a new driver might need to be created for NMEA 0183 protocol.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 01:59:47 AM
The last page of this document shows that "weird" format:

http://linux.geodatapub.com/shipwebpages/winfrog-all/WinFrog%203.10/Documents/WinFrog%203.10%20User%27s%20Guide/WFUG%20App%20C%20-%20Device%20Documentation/ROV/PerryTritec.pdf

The best thing you can do is provide all the details of your setup and devices.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 06, 2019, 02:14:12 AM
i have a terminal protocol software which was created in JAVA that can read any data format by assigning the format in initial place. unfortunately it's not in VB. you can see the attachment. my main target is actually to implement the same function into the AHMI.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 02:37:59 AM
Just by looking at config.txt file, this might have a simple solution in VB (this is just a wild guess).

What you should do is examine the exact data you are receiving and post it here.
Since your program is already communicating with a device it might just be a matter of deciphering the data.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 06, 2019, 03:24:54 AM
as shown in the attachment, my device input is from COM1 and output is COM6 to my AHMI. (the output format that you see on COM6 has been modified from the JAVA software setting as what i like it to be, so not to bother much. i can set the output to be exactly the same as the input).

so far no output turn out on the AHMI software that i created.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 04:25:16 AM
Is your Java software closed when you try to use AHMI?
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 04:36:48 AM
You need to understand that you are not providing enough details about your setup.

The first issue you need to resolve is the communication between AHMI and your device.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 06, 2019, 05:03:31 AM
Please see the attachment on my current setup.
As i mentioned, the reason i want to make the build in read/write serial port software into AHMI is to eliminate the JAVA PC.
Since now the only way i can see there is data is using the JAVA, that's why i use this setup.
the JAVA is always open in order to send/receive data.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 01:02:19 PM
Here is how I understand it:

AHMI normally uses one of its drivers to establish communication to a device and collect data from it.

In your case, there is no AHMI driver that can communicate with your device marked "device that sends the data".

The reading would have to be done through VB code and collected data would have to be manually sorted and provided to AHMI controls and also sent to Omron PLC (if that's your intention). So, in your code, when you open the serial port then you have to do the reading and not writing.

Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 06, 2019, 05:16:31 PM
If you do manage to read the data then examine it and compare to the Java data.

If you can communicate to the person that created the Java program then ask for more information on the data format.
Everything kind of looks like a standard serial protocol communication.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 12, 2019, 01:42:22 AM
Sorry for late reply. At the moment, i bypass the JAVA terminal exchange program and connect straight to the device (in this case it is a PC which run a JAVA program that has multiple data). i can read the data as can be seen on the attachment. so i have few questions:
1. is there a way i can read the updated data only in a single line.
2. can i separate the data into individual? i have tried someone code from https://www.electroniclinic.com/how-to-split-a-string-message-and-access-the-sensors-values/ (https://www.electroniclinic.com/how-to-split-a-string-message-and-access-the-sensors-values/). here is the code:
Code: [Select]
Imports System.IO
Imports System.IO.Ports

Public Class Form1

    Dim value1 As Integer

    Private Sub Form1_Load(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles MyBase.Load
        SerialPort1.Close()
        SerialPort1.PortName = "com15"
        SerialPort1.BaudRate = "9600"
        SerialPort1.DataBits = 8
        SerialPort1.Parity = Parity.None
        SerialPort1.StopBits = StopBits.One
        SerialPort1.Handshake = Handshake.None
        SerialPort1.Encoding = System.Text.Encoding.Default
        SerialPort1.Open()
    End Sub



    Private Sub Timer1_Tick(ByVal sender As System.Object, ByVal e As System.EventArgs) Handles Timer1.Tick
        Dim s As String

        s = TextBox1.Text + "," + "," + "," + ","

        Dim somestring() As String
        ' Split string based on comma
        somestring = s.Split(New Char() {","c})

        TextBox2.Text = somestring(0)
        ' value1 = Convert.ToDecimal(TextBox2.Text)
        TextBox3.Text = somestring(1)
        TextBox4.Text = somestring(2)
        TextBox1.Text = ""


    End Sub

    Private Sub DataReceived(ByVal sender As Object, ByVal e As SerialDataReceivedEventArgs) Handles SerialPort1.DataReceived
        Try
            Dim mydata As String = ""
            mydata = SerialPort1.ReadExisting()

            If TextBox1.InvokeRequired Then
                TextBox1.Invoke(DirectCast(Sub() TextBox1.Text &= mydata, MethodInvoker))
            Else
                TextBox1.Text &= mydata
            End If
        Catch ex As Exception
            MessageBox.Show(ex.Message)
        End Try
    End Sub
End Class
but it doesn't work.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 13, 2019, 02:02:35 PM
Can you provide specific answers for these 2 questions:

1) What is the exact device you are using (brand name, type and model)?
2) Besides for this device, will you be using any other devices?
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 16, 2019, 01:55:03 AM
I'm sorry maybe I didn't give full info.
1. the device is actually a desktop PC. it is having a ROV (remotely operated underwater vehicle) software made by JAVA.
    in the software, there are many input data such as altitude, depth, heading, pitch, roll, etc...
    the reason I choose this method to get the incoming data is because I don't have any other device that can send serial data at the moment.
2. yes I will but it will be the same situation. I may need to read the data from specific software such as TSS cable tracker.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 16, 2019, 03:56:28 PM
So, all together it looks like you want to use AHMI as a graphical interface for the simulation.

Maybe something similar to this:  https://www.youtube.com/watch?v=FRxoemOd5F0

From your picture, the data seems to be coming in this format:

"┴Yes   20:37:00 12/09/19 0.0  0.0    0.0  0.0  -90.0 -90.0 -20.0  (followed by 20+ spaces)"

If this is continuous format and it doesn't change much, besides for numbers contained within the string, then you could try using a trick to display it within a TextBox as a single line (so set the TextBox's Multiline property to False):

Code: [Select]
    Private Sub DataReceived(ByVal sender As Object, ByVal e As SerialDataReceivedEventArgs) Handles SerialPort1.DataReceived
        Try
            Dim mydata As String = SerialPort1.ReadExisting()

            If mydata.StartsWith("┴") Then
                'Trim all the trailing spaces and remove ┴ character from the string.
                Dim resultingString As String = mydata.TrimEnd(New Char() {" "c}).Substring(1)

                'Trim all the excess spaces from the string.
                Do
                    resultingString = resultingString.Replace("  ", " "c)
                Loop Until resultingString.IndexOf("  ") = -1

                'Compare the current text in the TextBox to the resultingString and replace it if different.
                'This will update the values only when they change.
                'This should be every second due to the clock value but might be different due to speed of communication.
                If TextBox1.Text <> resultingString Then
                    If TextBox1.InvokeRequired Then
                        TextBox1.Invoke(DirectCast(Sub() TextBox1.Text = resultingString, MethodInvoker))
                    Else
                        TextBox1.Text = resultingString
                    End If
                End If
            End If
        Catch ex As Exception
            MessageBox.Show(ex.Message)
        End Try
    End Sub

I didn't test the code above but it should theoretically give you a string that looks like this:

"Yes 20:37:00 12/09/19 0.0 0.0 0.0 0.0 -90.0 -90.0 -20.0"

Now you could split it and pass the values to AHMI controls, similar to this:

Code: [Select]
    Private Sub TextBox1_TextChanged(sender As Object, e As EventArgs) Handles TextBox1.TextChanged
        If Not String.IsNullOrEmpty(TextBox1.Text) Then
            Dim somestring() As String
            'Split string based on space
            somestring = TextBox1.Text.Split(New Char() {" "c})

            'This should give us the following substrings:
            'somestring(0) = Yes
            'somestring(1) = 20:37:00
            'somestring(2) = 12/09/19
            'somestring(3) = 0.0
            'somestring(4) = 0.0
            'somestring(5) = 0.0
            'somestring(6) = 0.0
            'somestring(7) = -90.0
            'somestring(8) = -90.0
            'somestring(9) = -20.0

            AttitudeIndicatorInstrumentControlHMI1.PitchAngle = CDbl(somestring(6))
            AttitudeIndicatorInstrumentControlHMI1.RollAngle = CDbl(somestring(7))
            AirSpeedIndicatorInstrumentControlHMI1.AirSpeed = CInt(somestring(3))
            AltimeterInstrumentControlHMI1.Altitude = CInt(somestring(5))
            HeadingIndicatorInstrumentControlHMI1.Heading = CInt(somestring(4))
            TurnCoordinatorInstrumentControlHMI1.TurnQuality = CSng(somestring(6))
            TurnCoordinatorInstrumentControlHMI1.TurnRate = CSng(somestring(9))
            VerticalSpeedIndicatorInstrumentControlHMI1.VerticalSpeed = CInt(somestring(9))
            'etc
        End If
    End Sub

I didn't test the code above either.

The whole code could skip a beat at certain times because of the serial communication, meaning how packets are received, or could even show different strings.


Resource:  https://stackoverflow.com/questions/26593520/scan-a-file-for-a-string-of-words-ignoring-extra-whitespaces-using-vb-net
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: Godra on September 17, 2019, 12:51:04 PM
If this works then you can eventually eliminate the TextBox and try to do all updating within the DataReceived sub, something like this:

Code: [Select]
    Private Sub DataReceived(ByVal sender As Object, ByVal e As SerialDataReceivedEventArgs) Handles SerialPort1.DataReceived
        Try
            Dim mydata As String = SerialPort1.ReadExisting()

            If mydata.StartsWith("┴") Then
                'Trim all the trailing spaces and remove ┴ character from the string.
                Dim resultingString As String = mydata.TrimEnd(New Char() {" "c}).Substring(1)

                'Trim all the excess spaces from the string.
                Do
                    resultingString = resultingString.Replace("  ", " "c)
                Loop Until resultingString.IndexOf("  ") = -1

                Dim somestring() As String
                'Split string based on space
                somestring = resultingString.Split(New Char() {" "c})

                'If Invoke is required then add it for each control (the same as it was done for the TextBox)
                AttitudeIndicatorInstrumentControlHMI1.PitchAngle = CDbl(somestring(6))
                AttitudeIndicatorInstrumentControlHMI1.RollAngle = CDbl(somestring(7))
                AirSpeedIndicatorInstrumentControlHMI1.AirSpeed = CInt(somestring(3))
                AltimeterInstrumentControlHMI1.Altitude = CInt(somestring(5))
                HeadingIndicatorInstrumentControlHMI1.Heading = CInt(somestring(4))
                TurnCoordinatorInstrumentControlHMI1.TurnQuality = CSng(somestring(6))
                TurnCoordinatorInstrumentControlHMI1.TurnRate = CSng(somestring(9))
                VerticalSpeedIndicatorInstrumentControlHMI1.VerticalSpeed = CInt(somestring(9))
            End If
        Catch ex As Exception
            MessageBox.Show(ex.Message)
        End Try
    End Sub

Just a possibility.
Title: Re: SEND PLC DATA TO OTHER COM PORT
Post by: joko markono on September 26, 2019, 04:27:20 AM
I'm truly sorry haven't had change to give a reply. i will try the code that you show and will let you know later.