happyDomain/forms/field.go

141 lines
4.6 KiB
Go

// Copyright or © or Copr. happyDNS (2020)
//
// contact@happydomain.org
//
// This software is a computer program whose purpose is to provide a modern
// interface to interact with DNS systems.
//
// This software is governed by the CeCILL license under French law and abiding
// by the rules of distribution of free software. You can use, modify and/or
// redistribute the software under the terms of the CeCILL license as
// circulated by CEA, CNRS and INRIA at the following URL
// "http://www.cecill.info".
//
// As a counterpart to the access to the source code and rights to copy, modify
// and redistribute granted by the license, users are provided only with a
// limited warranty and the software's author, the holder of the economic
// rights, and the successive licensors have only limited liability.
//
// In this respect, the user's attention is drawn to the risks associated with
// loading, using, modifying and/or developing or reproducing the software by
// the user in light of its specific status of free software, that may mean
// that it is complicated to manipulate, and that also therefore means that it
// is reserved for developers and experienced professionals having in-depth
// computer knowledge. Users are therefore encouraged to load and test the
// software's suitability as regards their requirements in conditions enabling
// the security of their systems and/or data to be ensured and, more generally,
// to use and operate it in the same conditions as regards security.
//
// The fact that you are presently reading this means that you have had
// knowledge of the CeCILL license and that you accept its terms.
package forms // import "happydns.org/forms"
import (
"reflect"
"strings"
)
// Field
type Field struct {
// Id is the field identifier.
Id string `json:"id"`
// Type is the string representation of the field's type.
Type string `json:"type"`
// Label is the title given to the field, displayed as <label> tag on the interface.
Label string `json:"label,omitempty"`
// Placeholder is the placeholder attribute of the corresponding <input> tag.
Placeholder string `json:"placeholder,omitempty"`
// Default is the preselected value or the default value in case the field is not filled by the user.
Default string `json:"default,omitempty"`
// Choices holds the differents choices shown to the user in <select> tag.
Choices []string `json:"choices,omitempty"`
// Required indicates whether the field has to be filled or not.
Required bool `json:"required,omitempty"`
// Secret indicates if the field contains sensitive information such as API key, in order to hide
// the field when not needed. When typing, it doesn't hide characters like in password input.
Secret bool `json:"secret,omitempty"`
// Description stores an helpfull sentence describing the field.
Description string `json:"description,omitempty"`
}
// GenField generates a generic Field based on the happydomain tag.
func GenField(field reflect.StructField) (f *Field) {
jsonTag := field.Tag.Get("json")
jsonTuples := strings.Split(jsonTag, ",")
f = &Field{
Type: field.Type.String(),
}
if len(jsonTuples) > 0 && len(jsonTuples[0]) > 0 {
f.Id = jsonTuples[0]
} else {
f.Id = field.Name
}
tag := field.Tag.Get("happydomain")
tuples := strings.Split(tag, ",")
for _, t := range tuples {
kv := strings.SplitN(t, "=", 2)
if len(kv) > 1 {
switch strings.ToLower(kv[0]) {
case "label":
f.Label = kv[1]
case "placeholder":
f.Placeholder = kv[1]
case "default":
f.Default = kv[1]
case "description":
f.Description = kv[1]
case "choices":
f.Choices = strings.Split(kv[1], ";")
}
} else {
switch strings.ToLower(kv[0]) {
case "required":
f.Required = true
case "secret":
f.Secret = true
default:
f.Label = kv[0]
}
}
}
return
}
// GenStructFields generates corresponding SourceFields of the given Source.
func GenStructFields(data interface{}) (fields []*Field) {
if data != nil {
dataMeta := reflect.Indirect(reflect.ValueOf(data)).Type()
for i := 0; i < dataMeta.NumField(); i += 1 {
fields = append(fields, GenField(dataMeta.Field(i)))
}
}
return
}
// GenDefaultSettingsForm generates a generic CustomForm presenting all the fields in one page.
func GenDefaultSettingsForm(data interface{}) *CustomForm {
return &CustomForm{
Fields: GenStructFields(data),
NextButtonText: "common.create-thing",
NextEditButtonText: "common.update",
NextButtonState: 1,
PreviousButtonText: "provider.another",
PreviousEditButtonText: "common.cancel",
PreviousButtonState: -1,
}
}