module_index.go 9.7 KB

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