//
// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements.  See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership.  The ASF licenses this file
// to you under the Apache License, Version 2.0 (the
// "License"); you may not use this file except in compliance
// with the License.  You may obtain a copy of the License at
//
//      http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing,
// software distributed under the License is distributed on an
// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied.  See the License for the
// specific language governing permissions and limitations
// under the License.
//

package model

import (
	"github.com/apache/plc4x/plc4go/internal/plc4go/spi/utils"
)

// Code generated by build-utils. DO NOT EDIT.

// The data-structure of this message
type S7MessageUserData struct {
	Parent *S7Message
}

// The corresponding interface
type IS7MessageUserData interface {
	LengthInBytes() uint16
	LengthInBits() uint16
	Serialize(writeBuffer utils.WriteBuffer) error
}

///////////////////////////////////////////////////////////
// Accessors for discriminator values.
///////////////////////////////////////////////////////////
func (m *S7MessageUserData) MessageType() uint8 {
	return 0x07
}

func (m *S7MessageUserData) InitializeParent(parent *S7Message, tpduReference uint16, parameter *S7Parameter, payload *S7Payload) {
	m.Parent.TpduReference = tpduReference
	m.Parent.Parameter = parameter
	m.Parent.Payload = payload
}

func NewS7MessageUserData(tpduReference uint16, parameter *S7Parameter, payload *S7Payload) *S7Message {
	child := &S7MessageUserData{
		Parent: NewS7Message(tpduReference, parameter, payload),
	}
	child.Parent.Child = child
	return child.Parent
}

func CastS7MessageUserData(structType interface{}) *S7MessageUserData {
	castFunc := func(typ interface{}) *S7MessageUserData {
		if casted, ok := typ.(S7MessageUserData); ok {
			return &casted
		}
		if casted, ok := typ.(*S7MessageUserData); ok {
			return casted
		}
		if casted, ok := typ.(S7Message); ok {
			return CastS7MessageUserData(casted.Child)
		}
		if casted, ok := typ.(*S7Message); ok {
			return CastS7MessageUserData(casted.Child)
		}
		return nil
	}
	return castFunc(structType)
}

func (m *S7MessageUserData) GetTypeName() string {
	return "S7MessageUserData"
}

func (m *S7MessageUserData) LengthInBits() uint16 {
	return m.LengthInBitsConditional(false)
}

func (m *S7MessageUserData) LengthInBitsConditional(lastItem bool) uint16 {
	lengthInBits := uint16(m.Parent.ParentLengthInBits())

	return lengthInBits
}

func (m *S7MessageUserData) LengthInBytes() uint16 {
	return m.LengthInBits() / 8
}

func S7MessageUserDataParse(readBuffer utils.ReadBuffer) (*S7Message, error) {
	if pullErr := readBuffer.PullContext("S7MessageUserData"); pullErr != nil {
		return nil, pullErr
	}

	if closeErr := readBuffer.CloseContext("S7MessageUserData"); closeErr != nil {
		return nil, closeErr
	}

	// Create a partially initialized instance
	_child := &S7MessageUserData{
		Parent: &S7Message{},
	}
	_child.Parent.Child = _child
	return _child.Parent, nil
}

func (m *S7MessageUserData) Serialize(writeBuffer utils.WriteBuffer) error {
	ser := func() error {
		if pushErr := writeBuffer.PushContext("S7MessageUserData"); pushErr != nil {
			return pushErr
		}

		if popErr := writeBuffer.PopContext("S7MessageUserData"); popErr != nil {
			return popErr
		}
		return nil
	}
	return m.Parent.SerializeParent(writeBuffer, m, ser)
}

func (m *S7MessageUserData) String() string {
	if m == nil {
		return "<nil>"
	}
	buffer := utils.NewBoxedWriteBufferWithOptions(true, true)
	m.Serialize(buffer)
	return buffer.GetBox().String()
}
