with a label.
// IMPORTANT:
// The `fieldName` argument will cause a panic if it is not exactly the string
// form of the struct field that this editor input is representing
func Checkbox(fieldName string, p interface{}, attrs, options map[string]string) []byte {
attrs["class"] = "input-field col s12"
div := newElement("div", attrs["label"], fieldName, p, attrs)
var opts []*element
// get the pre-checked options if this is already an existing post
checkedVals := valueFromStructField(fieldName, p)
checked := strings.Split(checkedVals, "__ponzu")
i := 0
for k, v := range options {
inputAttrs := map[string]string{
"type": "checkbox",
"value": k,
"id": strings.Join(strings.Split(v, " "), "-"),
}
// check if k is in the pre-checked values and set to checked
for _, x := range checked {
if k == x {
inputAttrs["checked"] = "checked"
}
}
// create a *element manually using the maodified tagNameFromStructFieldMulti
// func since this is for a multi-value name
input := &element{
TagName: "input",
Attrs: inputAttrs,
Name: tagNameFromStructFieldMulti(fieldName, i, p),
label: v,
data: "",
viewBuf: &bytes.Buffer{},
}
opts = append(opts, input)
i++
}
return domElementWithChildrenCheckbox(div, opts)
}
// Tags returns the []byte of a tag input (in the style of Materialze 'Chips') with a label.
// IMPORTANT:
// The `fieldName` argument will cause a panic if it is not exactly the string
// form of the struct field that this editor input is representing
func Tags(fieldName string, p interface{}, attrs map[string]string) []byte {
name := tagNameFromStructField(fieldName, p)
// get the saved tags if this is already an existing post
values := valueFromStructField(fieldName, p)
var tags []string
if strings.Contains(values, "__ponzu") {
tags = strings.Split(values, "__ponzu")
}
// case where there is only one tag stored, thus has no separator
if len(values) > 0 && !strings.Contains(values, "__ponzu") {
tags = append(tags, values)
}
html := `
`
return []byte(html + script)
}
// domElementSelfClose is a special DOM element which is parsed as a
// self-closing tag and thus needs to be created differently
func domElementSelfClose(e *element) []byte {
e.viewBuf.Write([]byte(`
`))
if e.label != "" {
e.viewBuf.Write([]byte(`` + e.label + ` `))
}
e.viewBuf.Write([]byte(`<` + e.TagName + ` value="`))
e.viewBuf.Write([]byte(html.EscapeString(e.data) + `" `))
for attr, value := range e.Attrs {
e.viewBuf.Write([]byte(attr + `="` + value + `" `))
}
e.viewBuf.Write([]byte(` name="` + e.Name + `"`))
e.viewBuf.Write([]byte(` />`))
e.viewBuf.Write([]byte(`
`))
return e.viewBuf.Bytes()
}
// domElementCheckbox is a special DOM element which is parsed as a
// checkbox input tag and thus needs to be created differently
func domElementCheckbox(e *element) []byte {
e.viewBuf.Write([]byte(`
`))
e.viewBuf.Write([]byte(`<` + e.TagName + ` `))
for attr, value := range e.Attrs {
e.viewBuf.Write([]byte(attr + `="` + value + `" `))
}
e.viewBuf.Write([]byte(` name="` + e.Name + `"`))
e.viewBuf.Write([]byte(` /> `))
if e.label != "" {
e.viewBuf.Write([]byte(`` + e.label + ` `))
}
e.viewBuf.Write([]byte(`
`))
return e.viewBuf.Bytes()
}
// domElement creates a DOM element
func domElement(e *element) []byte {
e.viewBuf.Write([]byte(`
`))
if e.label != "" {
e.viewBuf.Write([]byte(`` + e.label + ` `))
}
e.viewBuf.Write([]byte(`<` + e.TagName + ` `))
for attr, value := range e.Attrs {
e.viewBuf.Write([]byte(attr + `="` + string(value) + `" `))
}
e.viewBuf.Write([]byte(` name="` + e.Name + `"`))
e.viewBuf.Write([]byte(` >`))
e.viewBuf.Write([]byte(html.EscapeString(e.data)))
e.viewBuf.Write([]byte(`` + e.TagName + `>`))
e.viewBuf.Write([]byte(`
`))
return e.viewBuf.Bytes()
}
func domElementWithChildrenSelect(e *element, children []*element) []byte {
e.viewBuf.Write([]byte(`
`))
e.viewBuf.Write([]byte(`<` + e.TagName + ` `))
for attr, value := range e.Attrs {
e.viewBuf.Write([]byte(attr + `="` + string(value) + `" `))
}
e.viewBuf.Write([]byte(` name="` + e.Name + `"`))
e.viewBuf.Write([]byte(` >`))
// loop over children and create domElement for each child
for _, child := range children {
e.viewBuf.Write(domElement(child))
}
e.viewBuf.Write([]byte(`` + e.TagName + `>`))
if e.label != "" {
e.viewBuf.Write([]byte(`` + e.label + ` `))
}
e.viewBuf.Write([]byte(`
`))
return e.viewBuf.Bytes()
}
func domElementWithChildrenCheckbox(e *element, children []*element) []byte {
e.viewBuf.Write([]byte(`<` + e.TagName + ` `))
for attr, value := range e.Attrs {
e.viewBuf.Write([]byte(attr + `="` + value + `" `))
}
e.viewBuf.Write([]byte(` >`))
if e.label != "" {
e.viewBuf.Write([]byte(`
` + e.label + ` `))
}
// loop over children and create domElement for each child
for _, child := range children {
e.viewBuf.Write(domElementCheckbox(child))
}
e.viewBuf.Write([]byte(`` + e.TagName + `>
`))
return e.viewBuf.Bytes()
}
func tagNameFromStructField(name string, post interface{}) string {
// sometimes elements in these environments will not have a name,
// and thus no tag name in the struct which correlates to it.
if name == "" {
return name
}
field, ok := reflect.TypeOf(post).Elem().FieldByName(name)
if !ok {
panic("Couldn't get struct field for: " + name + ". Make sure you pass the right field name to editor field elements.")
}
tag, ok := field.Tag.Lookup("json")
if !ok {
panic("Couldn't get json struct tag for: " + name + ". Struct fields for content types must have 'json' tags.")
}
return tag
}
// due to the format in which gorilla/schema expects form names to be when
// one is associated with multiple values, we need to output the name as such.
// Ex. 'category.0', 'category.1', 'category.2' and so on.
func tagNameFromStructFieldMulti(name string, i int, post interface{}) string {
tag := tagNameFromStructField(name, post)
return fmt.Sprintf("%s.%d", tag, i)
}
func valueFromStructField(name string, post interface{}) string {
field := reflect.Indirect(reflect.ValueOf(post)).FieldByName(name)
switch field.Kind() {
case reflect.String:
return field.String()
case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
return fmt.Sprintf("%v", field.Int())
case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64:
return fmt.Sprintf("%v", field.Uint())
case reflect.Bool:
return fmt.Sprintf("%t", field.Bool())
case reflect.Complex64, reflect.Complex128:
return fmt.Sprintf("%v", field.Complex())
case reflect.Float32, reflect.Float64:
return fmt.Sprintf("%v", field.Float())
case reflect.Slice:
s := []string{}
for i := 0; i < field.Len(); i++ {
pos := field.Index(i)
s = append(s, fmt.Sprintf("%v", pos))
}
return strings.Join(s, "__ponzu")
default:
panic(fmt.Sprintf("Ponzu: Type '%s' for field '%s' not supported.", field.Type(), name))
}
}
func newElement(tagName, label, fieldName string, p interface{}, attrs map[string]string) *element {
return &element{
TagName: tagName,
Attrs: attrs,
Name: tagNameFromStructField(fieldName, p),
label: label,
data: valueFromStructField(fieldName, p),
viewBuf: &bytes.Buffer{},
}
}