module_index.go 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  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. err := wrap.DB.QueryRow(`
  28. SELECT
  29. id,
  30. user,
  31. name,
  32. alias,
  33. content,
  34. meta_title,
  35. meta_keywords,
  36. meta_description,
  37. UNIX_TIMESTAMP(datetime) as datetime,
  38. active
  39. FROM
  40. pages
  41. WHERE
  42. active = 1 and
  43. alias = ?
  44. LIMIT 1;`,
  45. wrap.R.URL.Path,
  46. ).Scan(
  47. &row.A_id,
  48. &row.A_user,
  49. &row.A_name,
  50. &row.A_alias,
  51. &row.A_content,
  52. &row.A_meta_title,
  53. &row.A_meta_keywords,
  54. &row.A_meta_description,
  55. &row.A_datetime,
  56. &row.A_active,
  57. )
  58. if err != nil && err != wrapper.ErrNoRows {
  59. // System error 500
  60. utils.SystemErrorPageEngine(wrap.W, err)
  61. return
  62. } else if err == wrapper.ErrNoRows {
  63. // User error 404 page
  64. wrap.RenderFrontEnd("404", fetdata.New(wrap, nil, true), http.StatusNotFound)
  65. return
  66. }
  67. // Replace title with page name
  68. if row.A_meta_title == "" {
  69. row.A_meta_title = row.A_name
  70. }
  71. // Which template
  72. tmpl_name := "index"
  73. if wrap.R.URL.Path != "/" {
  74. tmpl_name = "page"
  75. }
  76. // Render template
  77. wrap.RenderFrontEnd(tmpl_name, fetdata.New(wrap, row, false), http.StatusOK)
  78. }, func(wrap *wrapper.Wrapper) (string, string, string) {
  79. content := ""
  80. sidebar := ""
  81. if wrap.CurrSubModule == "" || wrap.CurrSubModule == "default" {
  82. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  83. {Name: "List of pages"},
  84. })
  85. content += builder.DataTable(
  86. wrap,
  87. "pages",
  88. "id",
  89. "DESC",
  90. &[]builder.DataTableRow{
  91. {
  92. DBField: "id",
  93. },
  94. {
  95. DBField: "name",
  96. NameInTable: "Page / URL",
  97. CallBack: func(values *[]string) string {
  98. name := `<a href="/cp/` + wrap.CurrModule + `/modify/` + (*values)[0] + `/">` + html.EscapeString((*values)[1]) + `</a>`
  99. alias := html.EscapeString((*values)[2])
  100. return `<div>` + name + `</div><div><small>` + alias + `</small></div>`
  101. },
  102. },
  103. {
  104. DBField: "alias",
  105. },
  106. {
  107. DBField: "datetime",
  108. DBExp: "UNIX_TIMESTAMP(`datetime`)",
  109. NameInTable: "Date / Time",
  110. Classes: "d-none d-md-table-cell",
  111. CallBack: func(values *[]string) string {
  112. t := int64(utils.StrToInt((*values)[3]))
  113. return `<div>` + utils.UnixTimestampToFormat(t, "02.01.2006") + `</div>` +
  114. `<div><small>` + utils.UnixTimestampToFormat(t, "15:04:05") + `</small></div>`
  115. },
  116. },
  117. {
  118. DBField: "active",
  119. NameInTable: "Active",
  120. Classes: "d-none d-sm-table-cell",
  121. CallBack: func(values *[]string) string {
  122. return builder.CheckBox(utils.StrToInt((*values)[4]))
  123. },
  124. },
  125. },
  126. func(values *[]string) string {
  127. return builder.DataTableAction(&[]builder.DataTableActionRow{
  128. {
  129. Icon: assets.SysSvgIconView,
  130. Href: (*values)[2],
  131. Hint: "View",
  132. Target: "_blank",
  133. },
  134. {
  135. Icon: assets.SysSvgIconEdit,
  136. Href: "/cp/" + wrap.CurrModule + "/modify/" + (*values)[0] + "/",
  137. Hint: "Edit",
  138. },
  139. {
  140. Icon: assets.SysSvgIconRemove,
  141. Href: "javascript:fave.ActionDataTableDelete(this,'index-delete','" +
  142. (*values)[0] + "','Are you sure want to delete page?');",
  143. Hint: "Delete",
  144. Classes: "delete",
  145. },
  146. })
  147. },
  148. "/cp/"+wrap.CurrModule+"/",
  149. nil,
  150. nil,
  151. true,
  152. )
  153. } else if wrap.CurrSubModule == "add" || wrap.CurrSubModule == "modify" {
  154. if wrap.CurrSubModule == "add" {
  155. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  156. {Name: "Add new page"},
  157. })
  158. } else {
  159. content += this.getBreadCrumbs(wrap, &[]consts.BreadCrumb{
  160. {Name: "Modify page"},
  161. })
  162. }
  163. data := utils.MySql_page{
  164. A_id: 0,
  165. A_user: 0,
  166. A_name: "",
  167. A_alias: "",
  168. A_content: "",
  169. A_meta_title: "",
  170. A_meta_keywords: "",
  171. A_meta_description: "",
  172. A_datetime: 0,
  173. A_active: 0,
  174. }
  175. if wrap.CurrSubModule == "modify" {
  176. if len(wrap.UrlArgs) != 3 {
  177. return "", "", ""
  178. }
  179. if !utils.IsNumeric(wrap.UrlArgs[2]) {
  180. return "", "", ""
  181. }
  182. err := wrap.DB.QueryRow(`
  183. SELECT
  184. id,
  185. user,
  186. name,
  187. alias,
  188. content,
  189. meta_title,
  190. meta_keywords,
  191. meta_description,
  192. active
  193. FROM
  194. pages
  195. WHERE
  196. id = ?
  197. LIMIT 1;`,
  198. utils.StrToInt(wrap.UrlArgs[2]),
  199. ).Scan(
  200. &data.A_id,
  201. &data.A_user,
  202. &data.A_name,
  203. &data.A_alias,
  204. &data.A_content,
  205. &data.A_meta_title,
  206. &data.A_meta_keywords,
  207. &data.A_meta_description,
  208. &data.A_active,
  209. )
  210. if err != nil {
  211. return "", "", ""
  212. }
  213. }
  214. btn_caption := "Add"
  215. if wrap.CurrSubModule == "modify" {
  216. btn_caption = "Save"
  217. }
  218. content += builder.DataForm(wrap, []builder.DataFormField{
  219. {
  220. Kind: builder.DFKHidden,
  221. Name: "action",
  222. Value: "index-modify",
  223. },
  224. {
  225. Kind: builder.DFKHidden,
  226. Name: "id",
  227. Value: utils.IntToStr(data.A_id),
  228. },
  229. {
  230. Kind: builder.DFKText,
  231. Caption: "Page name",
  232. Name: "name",
  233. Value: data.A_name,
  234. },
  235. {
  236. Kind: builder.DFKText,
  237. Caption: "Page alias",
  238. Name: "alias",
  239. Value: data.A_alias,
  240. Hint: "Example: /about-us/ or /about-us.html",
  241. },
  242. {
  243. Kind: builder.DFKTextArea,
  244. Caption: "Page content",
  245. Name: "content",
  246. Value: data.A_content,
  247. Classes: "autosize",
  248. },
  249. {
  250. Kind: builder.DFKText,
  251. Caption: "Meta title",
  252. Name: "meta_title",
  253. Value: data.A_meta_title,
  254. },
  255. {
  256. Kind: builder.DFKText,
  257. Caption: "Meta keywords",
  258. Name: "meta_keywords",
  259. Value: data.A_meta_keywords,
  260. },
  261. {
  262. Kind: builder.DFKTextArea,
  263. Caption: "Meta description",
  264. Name: "meta_description",
  265. Value: data.A_meta_description,
  266. },
  267. {
  268. Kind: builder.DFKCheckBox,
  269. Caption: "Active",
  270. Name: "active",
  271. Value: utils.IntToStr(data.A_active),
  272. },
  273. {
  274. Kind: builder.DFKMessage,
  275. },
  276. {
  277. Kind: builder.DFKSubmit,
  278. Value: btn_caption,
  279. Target: "add-edit-button",
  280. },
  281. })
  282. if wrap.CurrSubModule == "add" {
  283. sidebar += `<button class="btn btn-primary btn-sidebar" id="add-edit-button">Add</button>`
  284. } else {
  285. sidebar += `<button class="btn btn-primary btn-sidebar" id="add-edit-button">Save</button>`
  286. }
  287. }
  288. return this.getSidebarModules(wrap), content, sidebar
  289. })
  290. }
  291. func (this *Modules) RegisterAction_IndexModify() *Action {
  292. return this.newAction(AInfo{
  293. WantDB: true,
  294. Mount: "index-modify",
  295. WantAdmin: true,
  296. }, func(wrap *wrapper.Wrapper) {
  297. pf_id := wrap.R.FormValue("id")
  298. pf_name := wrap.R.FormValue("name")
  299. pf_alias := wrap.R.FormValue("alias")
  300. pf_content := wrap.R.FormValue("content")
  301. pf_meta_title := wrap.R.FormValue("meta_title")
  302. pf_meta_keywords := wrap.R.FormValue("meta_keywords")
  303. pf_meta_description := wrap.R.FormValue("meta_description")
  304. pf_active := wrap.R.FormValue("active")
  305. if pf_active == "" {
  306. pf_active = "0"
  307. }
  308. if !utils.IsNumeric(pf_id) {
  309. wrap.MsgError(`Inner system error`)
  310. return
  311. }
  312. if pf_name == "" {
  313. wrap.MsgError(`Please specify page name`)
  314. return
  315. }
  316. if pf_alias == "" {
  317. pf_alias = utils.GenerateAlias(pf_name)
  318. }
  319. if !utils.IsValidAlias(pf_alias) {
  320. wrap.MsgError(`Please specify correct page alias`)
  321. return
  322. }
  323. if pf_id == "0" {
  324. // Add new page
  325. _, err := wrap.DB.Exec(
  326. `INSERT INTO pages SET
  327. user = ?,
  328. name = ?,
  329. alias = ?,
  330. content = ?,
  331. meta_title = ?,
  332. meta_keywords = ?,
  333. meta_description = ?,
  334. datetime = ?,
  335. active = ?
  336. ;`,
  337. wrap.User.A_id,
  338. pf_name,
  339. pf_alias,
  340. pf_content,
  341. pf_meta_title,
  342. pf_meta_keywords,
  343. pf_meta_description,
  344. utils.UnixTimestampToMySqlDateTime(utils.GetCurrentUnixTimestamp()),
  345. pf_active,
  346. )
  347. if err != nil {
  348. wrap.MsgError(err.Error())
  349. return
  350. }
  351. wrap.Write(`window.location='/cp/';`)
  352. } else {
  353. // Update page
  354. _, err := wrap.DB.Exec(
  355. `UPDATE pages SET
  356. name = ?,
  357. alias = ?,
  358. content = ?,
  359. meta_title = ?,
  360. meta_keywords = ?,
  361. meta_description = ?,
  362. active = ?
  363. WHERE
  364. id = ?
  365. ;`,
  366. pf_name,
  367. pf_alias,
  368. pf_content,
  369. pf_meta_title,
  370. pf_meta_keywords,
  371. pf_meta_description,
  372. pf_active,
  373. utils.StrToInt(pf_id),
  374. )
  375. if err != nil {
  376. wrap.MsgError(err.Error())
  377. return
  378. }
  379. wrap.Write(`window.location='/cp/index/modify/` + pf_id + `/';`)
  380. }
  381. })
  382. }
  383. func (this *Modules) RegisterAction_IndexDelete() *Action {
  384. return this.newAction(AInfo{
  385. WantDB: true,
  386. Mount: "index-delete",
  387. WantAdmin: true,
  388. }, func(wrap *wrapper.Wrapper) {
  389. pf_id := wrap.R.FormValue("id")
  390. if !utils.IsNumeric(pf_id) {
  391. wrap.MsgError(`Inner system error`)
  392. return
  393. }
  394. err := wrap.DB.Transaction(func(tx *wrapper.Tx) error {
  395. // Process
  396. if _, err := tx.Exec("DELETE FROM pages WHERE id = ?;", pf_id); err != nil {
  397. return err
  398. }
  399. return nil
  400. })
  401. if err != nil {
  402. wrap.MsgError(err.Error())
  403. return
  404. }
  405. // Reload current page
  406. wrap.Write(`window.location.reload(false);`)
  407. })
  408. }