module_templates.go 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195
  1. package modules
  2. import (
  3. "html"
  4. "io/ioutil"
  5. "net/url"
  6. "os"
  7. "path/filepath"
  8. "strings"
  9. "golang-fave/engine/assets"
  10. "golang-fave/engine/builder"
  11. "golang-fave/engine/consts"
  12. "golang-fave/engine/utils"
  13. "golang-fave/engine/wrapper"
  14. )
  15. func (this *Modules) templates_GetThemeFiles(wrap *wrapper.Wrapper) []string {
  16. var result []string
  17. files, err := ioutil.ReadDir(wrap.DTemplate)
  18. if err == nil {
  19. for _, file := range files {
  20. if len(file.Name()) > 0 && file.Name()[0] == '.' {
  21. continue
  22. }
  23. if len(file.Name()) > 0 && strings.ToLower(file.Name()) == "robots.txt" {
  24. continue
  25. }
  26. result = append(result, file.Name())
  27. }
  28. }
  29. return result
  30. }
  31. func (this *Modules) RegisterModule_Templates() *Module {
  32. return this.newModule(MInfo{
  33. Mount: "templates",
  34. Name: "Templates",
  35. Order: 802,
  36. System: true,
  37. Icon: assets.SysSvgIconView,
  38. Sub: &[]MISub{
  39. {Mount: "default", Name: "Template editor", Show: true, Icon: assets.SysSvgIconEdit},
  40. {Mount: "create", Name: "Add new template", Show: true, Icon: assets.SysSvgIconPlus},
  41. {Mount: "restore", Name: "Restore", Show: true, Icon: assets.SysSvgIconRestore},
  42. },
  43. }, nil, func(wrap *wrapper.Wrapper) (string, string, string) {
  44. content := ""
  45. sidebar := ""
  46. if wrap.CurrSubModule == "" || wrap.CurrSubModule == "default" {
  47. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  48. {Name: "Template editor"},
  49. })
  50. files := this.templates_GetThemeFiles(wrap)
  51. if len(files) > 0 {
  52. selected_file, _ := url.QueryUnescape(wrap.R.URL.Query().Get("file"))
  53. if !(selected_file != "" && utils.InArrayString(files, selected_file)) {
  54. selected_file = files[0]
  55. }
  56. list_of_files := ``
  57. for _, file := range files {
  58. selected := ""
  59. if file == selected_file {
  60. selected = " selected"
  61. }
  62. list_of_files += `<option value="` + html.EscapeString(file) +
  63. `"` + selected + `>` + html.EscapeString(file) + `</option>`
  64. }
  65. fcont := []byte(``)
  66. fcont, _ = ioutil.ReadFile(wrap.DTemplate + string(os.PathSeparator) + selected_file)
  67. fext := filepath.Ext(selected_file)
  68. if len(fext) > 2 {
  69. fext = fext[1:]
  70. }
  71. content += builder.DataForm(wrap, []builder.DataFormField{
  72. {
  73. Kind: builder.DFKHidden,
  74. Name: "action",
  75. Value: "templates-edit-theme-file",
  76. },
  77. {
  78. Kind: builder.DFKText,
  79. Caption: "Theme file",
  80. Name: "file",
  81. Value: "0",
  82. CallBack: func(field *builder.DataFormField) string {
  83. return `<div class="form-group n1">` +
  84. `<div class="row">` +
  85. `<div class="col-12">` +
  86. `<div style="position:relative;">` +
  87. `<button type="button" class="btn btn-success" onclick="return fave.ActionRestoreThemeFile('templates-restore-file','` + selected_file + `','Are you sure want to restore theme file?');" style="position:absolute;right:0;">Restore</button>` +
  88. `<select class="form-control ignore-lost-data" id="lbl_file" name="file" onchange="setTimeout(function(){$('#lbl_file').val('` + selected_file + `')},500);document.location='/cp/` + wrap.CurrModule + `/?file='+encodeURI(this.value);">` +
  89. list_of_files +
  90. `</select>` +
  91. `</div>` +
  92. `</div>` +
  93. `</div>` +
  94. `</div>`
  95. },
  96. },
  97. {
  98. Kind: builder.DFKText,
  99. CallBack: func(field *builder.DataFormField) string {
  100. return `<div class="form-group last"><div class="row"><div class="col-12"><textarea class="form-control tmpl-editor" name="content" data-emode="` + fext + `" placeholder="" autocomplete="off">` + html.EscapeString(string(fcont)) + `</textarea></div></div></div>`
  101. },
  102. },
  103. {
  104. Kind: builder.DFKSubmit,
  105. CallBack: func(field *builder.DataFormField) string {
  106. return `<div class="row d-lg-none"><div class="col-12"><div class="pt-3"><button type="submit" class="btn btn-primary" data-target="add-edit-button">Save</button></div></div></div>`
  107. },
  108. },
  109. })
  110. sidebar += `<button class="btn btn-primary btn-sidebar" id="add-edit-button">Save</button>`
  111. } else {
  112. content += `<div class="sys-messages">
  113. <div class="alert alert-warning" role="alert">
  114. <strong>Error!</strong> No any file found in theme folder
  115. </div>
  116. </div>`
  117. }
  118. } else if wrap.CurrSubModule == "create" {
  119. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  120. {Name: "Add new template"},
  121. })
  122. content += builder.DataForm(wrap, []builder.DataFormField{
  123. {
  124. Kind: builder.DFKHidden,
  125. Name: "action",
  126. Value: "templates-create-theme-file",
  127. },
  128. {
  129. Kind: builder.DFKText,
  130. Caption: "Theme file",
  131. Name: "file",
  132. Value: "0",
  133. CallBack: func(field *builder.DataFormField) string {
  134. return `<div class="form-group n1">` +
  135. `<div class="row">` +
  136. `<div class="col-12">` +
  137. `<div style="position:relative;">` +
  138. `<input class="form-control ignore-lost-data" type="text" id="lbl_name" name="name" value="" minlength="1" maxlength="250" placeholder="New template file name without extension" autocomplete="off" required>` +
  139. `</div>` +
  140. `</div>` +
  141. `</div>` +
  142. `</div>`
  143. },
  144. },
  145. {
  146. Kind: builder.DFKText,
  147. CallBack: func(field *builder.DataFormField) string {
  148. return `<div class="form-group last"><div class="row"><div class="col-12"><textarea class="form-control tmpl-editor" name="content" data-emode="html" placeholder="" autocomplete="off"></textarea></div></div></div>`
  149. },
  150. },
  151. {
  152. Kind: builder.DFKSubmit,
  153. CallBack: func(field *builder.DataFormField) string {
  154. return `<div class="row d-lg-none"><div class="col-12"><div class="pt-3"><button type="submit" class="btn btn-primary" data-target="add-edit-button">Save</button></div></div></div>`
  155. },
  156. },
  157. })
  158. sidebar += `<button class="btn btn-primary btn-sidebar" id="add-edit-button">Add</button>`
  159. } else if wrap.CurrSubModule == "restore" {
  160. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  161. {Name: "Restore"},
  162. })
  163. content += builder.DataForm(wrap, []builder.DataFormField{
  164. {
  165. Kind: builder.DFKText,
  166. CallBack: func(field *builder.DataFormField) string {
  167. return `<div class="form-group last"><div class="row"><div class="col-12"><div class="alert alert-danger" style="margin:0;"><strong>WARNING!</strong><br>This action will restore current theme files to original, you will lost you theme changes!<br>Think twice before run this action! If you still want to do this, please press <b>Restore</b> red button!</div></div></div></div>`
  168. },
  169. },
  170. {
  171. Kind: builder.DFKSubmit,
  172. CallBack: func(field *builder.DataFormField) string {
  173. return `<div class="row d-lg-none"><div class="col-12"><div class="pt-3"><button type="button" class="btn btn-danger" onclick="return fave.ActionRestoreThemeFile('templates-restore-file-all','all','WARNING! Are you sure want to restore all theme files?');">Restore</button></div></div></div>`
  174. },
  175. },
  176. })
  177. sidebar += `<button class="btn btn-danger btn-sidebar" onclick="return fave.ActionRestoreThemeFile('templates-restore-file-all','all','WARNING! Are you sure want to restore all theme files?');" id="add-edit-button">Restore</button>`
  178. }
  179. return this.getSidebarModules(wrap), content, sidebar
  180. })
  181. }