// Code generated by go-swagger; DO NOT EDIT.

// Copyright 2018 The go-netbox Authors.
//
// Licensed 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 dcim

// This file was generated by the swagger tool.
// Editing this file might prove futile when you re-run the swagger generate command

import (
	"net/http"
	"time"

	"golang.org/x/net/context"

	"github.com/go-openapi/errors"
	"github.com/go-openapi/runtime"
	cr "github.com/go-openapi/runtime/client"

	strfmt "github.com/go-openapi/strfmt"

	"github.com/digitalocean/go-netbox/netbox/models"
)

// NewDcimPlatformsCreateParams creates a new DcimPlatformsCreateParams object
// with the default values initialized.
func NewDcimPlatformsCreateParams() *DcimPlatformsCreateParams {
	var ()
	return &DcimPlatformsCreateParams{

		timeout: cr.DefaultTimeout,
	}
}

// NewDcimPlatformsCreateParamsWithTimeout creates a new DcimPlatformsCreateParams object
// with the default values initialized, and the ability to set a timeout on a request
func NewDcimPlatformsCreateParamsWithTimeout(timeout time.Duration) *DcimPlatformsCreateParams {
	var ()
	return &DcimPlatformsCreateParams{

		timeout: timeout,
	}
}

// NewDcimPlatformsCreateParamsWithContext creates a new DcimPlatformsCreateParams object
// with the default values initialized, and the ability to set a context for a request
func NewDcimPlatformsCreateParamsWithContext(ctx context.Context) *DcimPlatformsCreateParams {
	var ()
	return &DcimPlatformsCreateParams{

		Context: ctx,
	}
}

// NewDcimPlatformsCreateParamsWithHTTPClient creates a new DcimPlatformsCreateParams object
// with the default values initialized, and the ability to set a custom HTTPClient for a request
func NewDcimPlatformsCreateParamsWithHTTPClient(client *http.Client) *DcimPlatformsCreateParams {
	var ()
	return &DcimPlatformsCreateParams{
		HTTPClient: client,
	}
}

/*DcimPlatformsCreateParams contains all the parameters to send to the API endpoint
for the dcim platforms create operation typically these are written to a http.Request
*/
type DcimPlatformsCreateParams struct {

	/*Data*/
	Data *models.WritablePlatform

	timeout    time.Duration
	Context    context.Context
	HTTPClient *http.Client
}

// WithTimeout adds the timeout to the dcim platforms create params
func (o *DcimPlatformsCreateParams) WithTimeout(timeout time.Duration) *DcimPlatformsCreateParams {
	o.SetTimeout(timeout)
	return o
}

// SetTimeout adds the timeout to the dcim platforms create params
func (o *DcimPlatformsCreateParams) SetTimeout(timeout time.Duration) {
	o.timeout = timeout
}

// WithContext adds the context to the dcim platforms create params
func (o *DcimPlatformsCreateParams) WithContext(ctx context.Context) *DcimPlatformsCreateParams {
	o.SetContext(ctx)
	return o
}

// SetContext adds the context to the dcim platforms create params
func (o *DcimPlatformsCreateParams) SetContext(ctx context.Context) {
	o.Context = ctx
}

// WithHTTPClient adds the HTTPClient to the dcim platforms create params
func (o *DcimPlatformsCreateParams) WithHTTPClient(client *http.Client) *DcimPlatformsCreateParams {
	o.SetHTTPClient(client)
	return o
}

// SetHTTPClient adds the HTTPClient to the dcim platforms create params
func (o *DcimPlatformsCreateParams) SetHTTPClient(client *http.Client) {
	o.HTTPClient = client
}

// WithData adds the data to the dcim platforms create params
func (o *DcimPlatformsCreateParams) WithData(data *models.WritablePlatform) *DcimPlatformsCreateParams {
	o.SetData(data)
	return o
}

// SetData adds the data to the dcim platforms create params
func (o *DcimPlatformsCreateParams) SetData(data *models.WritablePlatform) {
	o.Data = data
}

// WriteToRequest writes these params to a swagger request
func (o *DcimPlatformsCreateParams) WriteToRequest(r runtime.ClientRequest, reg strfmt.Registry) error {

	if err := r.SetTimeout(o.timeout); err != nil {
		return err
	}
	var res []error

	if o.Data != nil {
		if err := r.SetBodyParam(o.Data); err != nil {
			return err
		}
	}

	if len(res) > 0 {
		return errors.CompositeValidationError(res...)
	}
	return nil
}
