module_index.go 7.6 KB

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