module_index.go 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. package modules
  2. import (
  3. "html"
  4. "net/http"
  5. "golang-fave/assets"
  6. "golang-fave/consts"
  7. "golang-fave/engine/builder"
  8. "golang-fave/engine/fetdata"
  9. "golang-fave/engine/wrapper"
  10. "golang-fave/utils"
  11. )
  12. func (this *Modules) RegisterModule_Index() *Module {
  13. return this.newModule(MInfo{
  14. WantDB: true,
  15. Mount: "index",
  16. Name: "Pages",
  17. Order: 0,
  18. Icon: assets.SysSvgIconPage,
  19. Sub: &[]MISub{
  20. {Mount: "default", Name: "List of pages", Show: true, Icon: assets.SysSvgIconList},
  21. {Mount: "add", Name: "Add new page", Show: true, Icon: assets.SysSvgIconPlus},
  22. {Mount: "modify", Name: "Modify page", Show: false},
  23. },
  24. }, func(wrap *wrapper.Wrapper) {
  25. // Front-end
  26. row := &utils.MySql_page{}
  27. rou := &utils.MySql_user{}
  28. err := wrap.DB.QueryRow(`
  29. SELECT
  30. pages.id,
  31. pages.user,
  32. pages.name,
  33. pages.alias,
  34. pages.content,
  35. pages.meta_title,
  36. pages.meta_keywords,
  37. pages.meta_description,
  38. UNIX_TIMESTAMP(pages.datetime) as datetime,
  39. pages.active,
  40. users.id,
  41. users.first_name,
  42. users.last_name,
  43. users.email,
  44. users.admin,
  45. users.active
  46. FROM
  47. pages
  48. LEFT JOIN users ON users.id = pages.user
  49. WHERE
  50. pages.active = 1 and
  51. pages.alias = ?
  52. LIMIT 1;`,
  53. wrap.R.URL.Path,
  54. ).Scan(
  55. &row.A_id,
  56. &row.A_user,
  57. &row.A_name,
  58. &row.A_alias,
  59. &row.A_content,
  60. &row.A_meta_title,
  61. &row.A_meta_keywords,
  62. &row.A_meta_description,
  63. &row.A_datetime,
  64. &row.A_active,
  65. &rou.A_id,
  66. &rou.A_first_name,
  67. &rou.A_last_name,
  68. &rou.A_email,
  69. &rou.A_admin,
  70. &rou.A_active,
  71. )
  72. if err != nil && err != wrapper.ErrNoRows {
  73. // System error 500
  74. utils.SystemErrorPageEngine(wrap.W, err)
  75. return
  76. } else if err == wrapper.ErrNoRows {
  77. // User error 404 page
  78. wrap.RenderFrontEnd("404", fetdata.New(wrap, true, nil, nil), http.StatusNotFound)
  79. return
  80. }
  81. // Which template
  82. tname := "index"
  83. if wrap.R.URL.Path != "/" {
  84. tname = "page"
  85. }
  86. // Render template
  87. wrap.RenderFrontEnd(tname, fetdata.New(wrap, false, row, rou), http.StatusOK)
  88. }, func(wrap *wrapper.Wrapper) (string, string, string) {
  89. content := ""
  90. sidebar := ""
  91. if wrap.CurrSubModule == "" || wrap.CurrSubModule == "default" {
  92. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  93. {Name: "List of pages"},
  94. })
  95. content += builder.DataTable(
  96. wrap,
  97. "pages",
  98. "id",
  99. "DESC",
  100. &[]builder.DataTableRow{
  101. {
  102. DBField: "id",
  103. },
  104. {
  105. DBField: "name",
  106. NameInTable: "Page / URL",
  107. CallBack: func(values *[]string) string {
  108. name := `<a href="/cp/` + wrap.CurrModule + `/modify/` + (*values)[0] + `/">` + html.EscapeString((*values)[1]) + `</a>`
  109. alias := html.EscapeString((*values)[2])
  110. return `<div>` + name + `</div><div><small>` + alias + `</small></div>`
  111. },
  112. },
  113. {
  114. DBField: "alias",
  115. },
  116. {
  117. DBField: "datetime",
  118. DBExp: "UNIX_TIMESTAMP(`datetime`)",
  119. NameInTable: "Date / Time",
  120. Classes: "d-none d-md-table-cell",
  121. CallBack: func(values *[]string) string {
  122. t := int64(utils.StrToInt((*values)[3]))
  123. return `<div>` + utils.UnixTimestampToFormat(t, "02.01.2006") + `</div>` +
  124. `<div><small>` + utils.UnixTimestampToFormat(t, "15:04:05") + `</small></div>`
  125. },
  126. },
  127. {
  128. DBField: "active",
  129. NameInTable: "Active",
  130. Classes: "d-none d-sm-table-cell",
  131. CallBack: func(values *[]string) string {
  132. return builder.CheckBox(utils.StrToInt((*values)[4]))
  133. },
  134. },
  135. },
  136. func(values *[]string) string {
  137. return builder.DataTableAction(&[]builder.DataTableActionRow{
  138. {
  139. Icon: assets.SysSvgIconView,
  140. Href: (*values)[2],
  141. Hint: "View",
  142. Target: "_blank",
  143. },
  144. {
  145. Icon: assets.SysSvgIconEdit,
  146. Href: "/cp/" + wrap.CurrModule + "/modify/" + (*values)[0] + "/",
  147. Hint: "Edit",
  148. },
  149. {
  150. Icon: assets.SysSvgIconRemove,
  151. Href: "javascript:fave.ActionDataTableDelete(this,'index-delete','" +
  152. (*values)[0] + "','Are you sure want to delete page?');",
  153. Hint: "Delete",
  154. Classes: "delete",
  155. },
  156. })
  157. },
  158. "/cp/"+wrap.CurrModule+"/",
  159. nil,
  160. nil,
  161. true,
  162. )
  163. } else if wrap.CurrSubModule == "add" || wrap.CurrSubModule == "modify" {
  164. if wrap.CurrSubModule == "add" {
  165. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  166. {Name: "Add new page"},
  167. })
  168. } else {
  169. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  170. {Name: "Modify page"},
  171. })
  172. }
  173. data := utils.MySql_page{
  174. A_id: 0,
  175. A_user: 0,
  176. A_name: "",
  177. A_alias: "",
  178. A_content: "",
  179. A_meta_title: "",
  180. A_meta_keywords: "",
  181. A_meta_description: "",
  182. A_datetime: 0,
  183. A_active: 0,
  184. }
  185. if wrap.CurrSubModule == "modify" {
  186. if len(wrap.UrlArgs) != 3 {
  187. return "", "", ""
  188. }
  189. if !utils.IsNumeric(wrap.UrlArgs[2]) {
  190. return "", "", ""
  191. }
  192. err := wrap.DB.QueryRow(`
  193. SELECT
  194. id,
  195. user,
  196. name,
  197. alias,
  198. content,
  199. meta_title,
  200. meta_keywords,
  201. meta_description,
  202. active
  203. FROM
  204. pages
  205. WHERE
  206. id = ?
  207. LIMIT 1;`,
  208. utils.StrToInt(wrap.UrlArgs[2]),
  209. ).Scan(
  210. &data.A_id,
  211. &data.A_user,
  212. &data.A_name,
  213. &data.A_alias,
  214. &data.A_content,
  215. &data.A_meta_title,
  216. &data.A_meta_keywords,
  217. &data.A_meta_description,
  218. &data.A_active,
  219. )
  220. if err != nil {
  221. return "", "", ""
  222. }
  223. }
  224. btn_caption := "Add"
  225. if wrap.CurrSubModule == "modify" {
  226. btn_caption = "Save"
  227. }
  228. content += builder.DataForm(wrap, []builder.DataFormField{
  229. {
  230. Kind: builder.DFKHidden,
  231. Name: "action",
  232. Value: "index-modify",
  233. },
  234. {
  235. Kind: builder.DFKHidden,
  236. Name: "id",
  237. Value: utils.IntToStr(data.A_id),
  238. },
  239. {
  240. Kind: builder.DFKText,
  241. Caption: "Page name",
  242. Name: "name",
  243. Value: data.A_name,
  244. Required: true,
  245. Min: "1",
  246. Max: "255",
  247. },
  248. {
  249. Kind: builder.DFKText,
  250. Caption: "Page alias",
  251. Name: "alias",
  252. Value: data.A_alias,
  253. Hint: "Example: /about-us/ or /about-us.html",
  254. Max: "255",
  255. },
  256. {
  257. Kind: builder.DFKTextArea,
  258. Caption: "Page content",
  259. Name: "content",
  260. Value: data.A_content,
  261. Classes: "wysiwyg",
  262. },
  263. {
  264. Kind: builder.DFKText,
  265. Caption: "Meta title",
  266. Name: "meta_title",
  267. Value: data.A_meta_title,
  268. Max: "255",
  269. },
  270. {
  271. Kind: builder.DFKText,
  272. Caption: "Meta keywords",
  273. Name: "meta_keywords",
  274. Value: data.A_meta_keywords,
  275. Max: "255",
  276. },
  277. {
  278. Kind: builder.DFKTextArea,
  279. Caption: "Meta description",
  280. Name: "meta_description",
  281. Value: data.A_meta_description,
  282. Max: "510",
  283. },
  284. {
  285. Kind: builder.DFKCheckBox,
  286. Caption: "Active",
  287. Name: "active",
  288. Value: utils.IntToStr(data.A_active),
  289. },
  290. {
  291. Kind: builder.DFKSubmit,
  292. Value: btn_caption,
  293. Target: "add-edit-button",
  294. },
  295. })
  296. if wrap.CurrSubModule == "add" {
  297. sidebar += `<button class="btn btn-primary btn-sidebar" id="add-edit-button">Add</button>`
  298. } else {
  299. sidebar += `<button class="btn btn-primary btn-sidebar" id="add-edit-button">Save</button>`
  300. }
  301. }
  302. return this.getSidebarModules(wrap), content, sidebar
  303. })
  304. }