module_index.go 9.6 KB

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