Improved EditForm
This commit is contained in:
parent
8e62b31915
commit
1caa4b6de5
@ -31,7 +31,7 @@
|
|||||||
width 100%
|
width 100%
|
||||||
|
|
||||||
.widget-title
|
.widget-title
|
||||||
text-align left
|
horizontal
|
||||||
padding-bottom 0.5rem
|
padding-bottom 0.5rem
|
||||||
border-bottom 1px solid rgba(0, 0, 0, 0.1)
|
border-bottom 1px solid rgba(0, 0, 0, 0.1)
|
||||||
// We need !important here to overwrite the h3:first-child rule
|
// We need !important here to overwrite the h3:first-child rule
|
||||||
@ -56,4 +56,7 @@
|
|||||||
.widget-form
|
.widget-form
|
||||||
width 100%
|
width 100%
|
||||||
max-width 650px
|
max-width 650px
|
||||||
margin 0 auto
|
margin 0 auto
|
||||||
|
|
||||||
|
.indent
|
||||||
|
margin-left 1rem
|
@ -60,10 +60,16 @@ func Render(obj interface{}, title string, user *arn.User) string {
|
|||||||
return b.String()
|
return b.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
// RenderObject ...
|
// RenderObject renders the UI for the object into the bytes buffer and appends an ID prefix for all API requests.
|
||||||
|
// The ID prefix should either be empty or end with a dot character.
|
||||||
func RenderObject(b *bytes.Buffer, obj interface{}, idPrefix string) {
|
func RenderObject(b *bytes.Buffer, obj interface{}, idPrefix string) {
|
||||||
t := reflect.TypeOf(obj).Elem()
|
t := reflect.TypeOf(obj)
|
||||||
v := reflect.ValueOf(obj).Elem()
|
v := reflect.ValueOf(obj)
|
||||||
|
|
||||||
|
if t.Kind() == reflect.Ptr {
|
||||||
|
t = t.Elem()
|
||||||
|
v = v.Elem()
|
||||||
|
}
|
||||||
|
|
||||||
// Fields
|
// Fields
|
||||||
for i := 0; i < t.NumField(); i++ {
|
for i := 0; i < t.NumField(); i++ {
|
||||||
@ -79,9 +85,10 @@ func RenderField(b *bytes.Buffer, v *reflect.Value, field reflect.StructField, i
|
|||||||
}
|
}
|
||||||
|
|
||||||
fieldValue := reflect.Indirect(v.FieldByName(field.Name))
|
fieldValue := reflect.Indirect(v.FieldByName(field.Name))
|
||||||
|
fieldType := field.Type.String()
|
||||||
|
|
||||||
switch field.Type.String() {
|
// String
|
||||||
case "string":
|
if fieldType == "string" {
|
||||||
if field.Tag.Get("datalist") != "" {
|
if field.Tag.Get("datalist") != "" {
|
||||||
dataList := field.Tag.Get("datalist")
|
dataList := field.Tag.Get("datalist")
|
||||||
values := arn.DataLists[dataList]
|
values := arn.DataLists[dataList]
|
||||||
@ -91,16 +98,48 @@ func RenderField(b *bytes.Buffer, v *reflect.Value, field reflect.StructField, i
|
|||||||
} else {
|
} else {
|
||||||
b.WriteString(components.InputText(idPrefix+field.Name, fieldValue.String(), field.Name, field.Tag.Get("tooltip")))
|
b.WriteString(components.InputText(idPrefix+field.Name, fieldValue.String(), field.Name, field.Tag.Get("tooltip")))
|
||||||
}
|
}
|
||||||
case "[]string":
|
|
||||||
b.WriteString(components.InputTags(idPrefix+field.Name, fieldValue.Interface().([]string), field.Name, field.Tag.Get("tooltip")))
|
return
|
||||||
case "bool":
|
}
|
||||||
|
|
||||||
|
// Bool
|
||||||
|
if fieldType == "bool" {
|
||||||
if field.Name == "IsDraft" {
|
if field.Name == "IsDraft" {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
case "[]*arn.ExternalMedia", "[]*arn.Link":
|
|
||||||
|
// TODO: Render bool type
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Array of strings
|
||||||
|
if fieldType == "[]string" {
|
||||||
|
b.WriteString(components.InputTags(idPrefix+field.Name, fieldValue.Interface().([]string), field.Name, field.Tag.Get("tooltip")))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Any kind of array
|
||||||
|
if strings.HasPrefix(fieldType, "[]") {
|
||||||
|
b.WriteString(`<div class="widget-section">`)
|
||||||
|
b.WriteString(`<h3 class="widget-title">`)
|
||||||
|
b.WriteString(field.Name)
|
||||||
|
b.WriteString(`</h3>`)
|
||||||
|
|
||||||
for sliceIndex := 0; sliceIndex < fieldValue.Len(); sliceIndex++ {
|
for sliceIndex := 0; sliceIndex < fieldValue.Len(); sliceIndex++ {
|
||||||
b.WriteString(`<div class="widget-section">`)
|
b.WriteString(`<div class="widget-section">`)
|
||||||
b.WriteString(`<div class="widget-title">` + strconv.Itoa(sliceIndex+1) + ". " + field.Name + `</div>`)
|
|
||||||
|
b.WriteString(`<div class="widget-title">`)
|
||||||
|
|
||||||
|
// Title
|
||||||
|
b.WriteString(strconv.Itoa(sliceIndex+1) + ". " + field.Name)
|
||||||
|
b.WriteString(`<div class="spacer"></div>`)
|
||||||
|
|
||||||
|
// Remove button
|
||||||
|
b.WriteString(`<button class="action" data-action="arrayRemove" data-trigger="click" data-field="` + field.Name + `" data-index="`)
|
||||||
|
b.WriteString(strconv.Itoa(sliceIndex))
|
||||||
|
b.WriteString(`">` + utils.RawIcon("trash") + `</button>`)
|
||||||
|
|
||||||
|
b.WriteString(`</div>`)
|
||||||
|
|
||||||
arrayObj := fieldValue.Index(sliceIndex).Interface()
|
arrayObj := fieldValue.Index(sliceIndex).Interface()
|
||||||
arrayIDPrefix := fmt.Sprintf("%s[%d].", field.Name, sliceIndex)
|
arrayIDPrefix := fmt.Sprintf("%s[%d].", field.Name, sliceIndex)
|
||||||
@ -109,15 +148,10 @@ func RenderField(b *bytes.Buffer, v *reflect.Value, field reflect.StructField, i
|
|||||||
// Preview
|
// Preview
|
||||||
// elementValue := fieldValue.Index(sliceIndex)
|
// elementValue := fieldValue.Index(sliceIndex)
|
||||||
// RenderArrayElement(b, &elementValue)
|
// RenderArrayElement(b, &elementValue)
|
||||||
if field.Type.String() == "[]*arn.ExternalMedia" {
|
if fieldType == "[]*arn.ExternalMedia" {
|
||||||
b.WriteString(components.ExternalMedia(fieldValue.Index(sliceIndex).Interface().(*arn.ExternalMedia)))
|
b.WriteString(components.ExternalMedia(fieldValue.Index(sliceIndex).Interface().(*arn.ExternalMedia)))
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove button
|
|
||||||
b.WriteString(`<div class="buttons"><button class="action" data-action="arrayRemove" data-trigger="click" data-field="` + field.Name + `" data-index="`)
|
|
||||||
b.WriteString(strconv.Itoa(sliceIndex))
|
|
||||||
b.WriteString(`">` + utils.RawIcon("trash") + `</button></div>`)
|
|
||||||
|
|
||||||
b.WriteString(`</div>`)
|
b.WriteString(`</div>`)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -125,17 +159,18 @@ func RenderField(b *bytes.Buffer, v *reflect.Value, field reflect.StructField, i
|
|||||||
b.WriteString(`<button class="action" data-action="arrayAppend" data-trigger="click" data-field="` + field.Name + `">` + utils.Icon("plus") + `Add ` + field.Name + `</button>`)
|
b.WriteString(`<button class="action" data-action="arrayAppend" data-trigger="click" data-field="` + field.Name + `">` + utils.Icon("plus") + `Add ` + field.Name + `</button>`)
|
||||||
b.WriteString(`</div>`)
|
b.WriteString(`</div>`)
|
||||||
|
|
||||||
case "arn.CompanyName":
|
|
||||||
b.WriteString(`<div class="widget-section">`)
|
|
||||||
b.WriteString(`<div class="widget-title">` + field.Name + `</div>`)
|
|
||||||
|
|
||||||
for i := 0; i < field.Type.NumField(); i++ {
|
|
||||||
subField := field.Type.Field(i)
|
|
||||||
RenderField(b, &fieldValue, subField, field.Name+".")
|
|
||||||
}
|
|
||||||
|
|
||||||
b.WriteString(`</div>`)
|
b.WriteString(`</div>`)
|
||||||
default:
|
return
|
||||||
panic("No edit form implementation for " + idPrefix + field.Name + " with type " + field.Type.String())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Any custom field type will be recursively rendered via another RenderObject call
|
||||||
|
b.WriteString(`<div class="widget-section">`)
|
||||||
|
b.WriteString(`<h3 class="widget-title">` + field.Name + `</h3>`)
|
||||||
|
|
||||||
|
// Indent the fields
|
||||||
|
b.WriteString(`<div class="indent">`)
|
||||||
|
RenderObject(b, fieldValue.Interface(), field.Name+".")
|
||||||
|
b.WriteString(`</div>`)
|
||||||
|
|
||||||
|
b.WriteString(`</div>`)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user