package editform import ( "fmt" "reflect" "strconv" "strings" "github.com/aerogo/api" "github.com/animenotifier/arn" "github.com/animenotifier/notify.moe/components" "github.com/animenotifier/notify.moe/utils" ) // Render renders a generic editing UI for any kind of datatype that has an ID. func Render(obj interface{}, title string, user *arn.User) string { t := reflect.TypeOf(obj).Elem() v := reflect.ValueOf(obj).Elem() id := findMainID(t, v) lowerCaseTypeName := strings.ToLower(t.Name()) endpoint := `/api/` + lowerCaseTypeName + `/` + id.String() var b strings.Builder b.WriteString(`<div class="widget-form">`) b.WriteString(`<div class="widget" data-api="` + endpoint + `">`) // Title b.WriteString(`<h1 class="mountable">`) b.WriteString(title) b.WriteString(`</h1>`) // Render the object with its fields RenderObject(&b, obj, "") // Additional buttons when logged in if user != nil { b.WriteString(`<div class="buttons">`) // Publish button _, ok := t.FieldByName("IsDraft") if ok { isDraft := v.FieldByName("IsDraft").Interface().(bool) if isDraft { b.WriteString(`<div class="buttons"><button class="mountable action" data-action="publish" data-trigger="click">` + utils.Icon("share-alt") + `Publish</button></div>`) } } // Redownload button track, isSoundTrack := obj.(*arn.SoundTrack) if isSoundTrack && !track.IsDraft && track.HasMediaByService("Youtube") && track.File == "" && (user.Role == "editor" || user.Role == "admin") { b.WriteString(`<button class="mountable action" data-action="downloadSoundTrackFile" data-trigger="click" data-id="` + track.ID + `">` + utils.Icon("refresh") + `Redownload</button>`) } // Delete button _, isDeletable := obj.(api.Deletable) if isDeletable && (user.Role == "editor" || user.Role == "admin") { returnPath := "" switch lowerCaseTypeName { case "anime": returnPath = "/explore" case "company": returnPath = "/companies" default: returnPath = "/" + lowerCaseTypeName + "s" } b.WriteString(`<button class="mountable action" data-action="deleteObject" data-trigger="click" data-return-path="` + returnPath + `" data-confirm-type="` + lowerCaseTypeName + `">` + utils.Icon("trash") + `Delete</button>`) } b.WriteString(`</div>`) } b.WriteString("</div>") b.WriteString("</div>") return b.String() } // 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 *strings.Builder, obj interface{}, idPrefix string) { t := reflect.TypeOf(obj) v := reflect.ValueOf(obj) if t.Kind() == reflect.Ptr { t = t.Elem() v = v.Elem() } // Fields for i := 0; i < t.NumField(); i++ { field := t.Field(i) RenderField(b, &v, field, idPrefix) } } // RenderField ... func RenderField(b *strings.Builder, v *reflect.Value, field reflect.StructField, idPrefix string) { fieldValue := reflect.Indirect(v.FieldByName(field.Name)) // Embedded fields if field.Anonymous { RenderObject(b, fieldValue.Interface(), idPrefix) return } if field.Tag.Get("editable") != "true" { return } b.WriteString("<div class='mountable'>") defer b.WriteString("</div>") fieldType := field.Type.String() // String if fieldType == "string" { renderStringField(b, v, field, idPrefix, fieldValue) return } // Int if fieldType == "int" { b.WriteString(components.InputNumber(idPrefix+field.Name, float64(fieldValue.Int()), field.Name, field.Tag.Get("tooltip"), "", "", "1")) return } // Float if fieldType == "float64" { b.WriteString(components.InputNumber(idPrefix+field.Name, fieldValue.Float(), field.Name, field.Tag.Get("tooltip"), "", "", "")) return } // Bool if fieldType == "bool" { if field.Name == "IsDraft" { return } b.WriteString(components.InputBool(idPrefix+field.Name, fieldValue.Bool(), field.Name, field.Tag.Get("tooltip"))) 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, "[]") { renderSliceField(b, v, field, idPrefix, fieldType, fieldValue) return } // 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>`) } // String field func renderStringField(b *strings.Builder, v *reflect.Value, field reflect.StructField, idPrefix string, fieldValue reflect.Value) { idType := field.Tag.Get("idType") // Try to infer the ID type by the field name if idType == "" { switch field.Name { case "AnimeID": idType = "Anime" case "CharacterID": idType = "Character" } } showPreview := idType != "" && fieldValue.String() != "" if showPreview { b.WriteString("<div class='widget-section-with-preview'>") } // Input field if field.Tag.Get("datalist") != "" { dataList := field.Tag.Get("datalist") values := arn.DataLists[dataList] b.WriteString(components.InputSelection(idPrefix+field.Name, fieldValue.String(), field.Name, field.Tag.Get("tooltip"), values)) } else if field.Tag.Get("type") == "textarea" { b.WriteString(components.InputTextArea(idPrefix+field.Name, fieldValue.String(), field.Name, field.Tag.Get("tooltip"))) } else if field.Tag.Get("type") == "upload" { endpoint := field.Tag.Get("endpoint") id := v.FieldByName("ID").String() endpoint = strings.Replace(endpoint, ":id", id, 1) b.WriteString(components.InputFileUpload(idPrefix+field.Name, field.Name, field.Tag.Get("filetype"), endpoint)) } else { b.WriteString(components.InputText(idPrefix+field.Name, fieldValue.String(), field.Name, field.Tag.Get("tooltip"))) } if showPreview { b.WriteString("<div class='widget-section-preview'>") } // Preview switch idType { case "Anime": animeID := fieldValue.String() anime, err := arn.GetAnime(animeID) if err == nil { b.WriteString(components.EditFormImagePreview(anime.Link(), anime.ImageLink("small"), true, anime.Title.ByUser(nil))) } case "Character": characterID := fieldValue.String() character, err := arn.GetCharacter(characterID) if err == nil { b.WriteString(components.EditFormImagePreview(character.Link(), character.ImageLink("medium"), true, character.Name.ByUser(nil))) } case "": break default: fmt.Println("Error: Unknown idType tag: " + idType) } // Close preview tags if showPreview { b.WriteString("</div></div>") } } // Slice field func renderSliceField(b *strings.Builder, v *reflect.Value, field reflect.StructField, idPrefix string, fieldType string, fieldValue reflect.Value) { 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++ { b.WriteString(`<div class="widget-section">`) 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" title="Delete this ` + field.Name + `" 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() arrayIDPrefix := fmt.Sprintf("%s[%d].", field.Name, sliceIndex) RenderObject(b, arrayObj, arrayIDPrefix) // Preview // elementValue := fieldValue.Index(sliceIndex) // RenderArrayElement(b, &elementValue) if fieldType == "[]*arn.ExternalMedia" { b.WriteString(components.ExternalMedia(fieldValue.Index(sliceIndex).Interface().(*arn.ExternalMedia))) } b.WriteString(`</div>`) } b.WriteString(`<div class="buttons">`) 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>`) } // findMainID finds the main ID of the object. func findMainID(t reflect.Type, v reflect.Value) reflect.Value { idField := v.FieldByName("ID") if idField.IsValid() { return reflect.Indirect(idField) } for i := 0; i < t.NumField(); i++ { field := t.Field(i) if field.Tag.Get("mainID") == "true" { return reflect.Indirect(v.Field(i)) } } panic("Type " + t.Name() + " doesn't have a main ID!") }