modules.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427
  1. package modules
  2. import (
  3. "html"
  4. "html/template"
  5. "net/http"
  6. "reflect"
  7. "sort"
  8. "strings"
  9. "golang-fave/assets"
  10. "golang-fave/consts"
  11. "golang-fave/engine/wrapper"
  12. "golang-fave/utils"
  13. )
  14. type MISub struct {
  15. Mount string
  16. Name string
  17. Icon string
  18. Show bool
  19. Sep bool
  20. }
  21. type MInfo struct {
  22. Id string
  23. WantDB bool
  24. Mount string
  25. Name string
  26. Order int
  27. System bool
  28. Icon string
  29. Sub *[]MISub
  30. }
  31. type Module struct {
  32. Info MInfo
  33. Front func(wrap *wrapper.Wrapper)
  34. Back func(wrap *wrapper.Wrapper) (string, string, string)
  35. }
  36. type AInfo struct {
  37. Id string
  38. WantDB bool
  39. Mount string
  40. WantUser bool
  41. WantAdmin bool
  42. }
  43. type Action struct {
  44. Info AInfo
  45. Act func(wrap *wrapper.Wrapper)
  46. }
  47. type Modules struct {
  48. mods map[string]*Module
  49. acts map[string]*Action
  50. }
  51. func (this *Modules) load() {
  52. t := reflect.TypeOf(this)
  53. for i := 0; i < t.NumMethod(); i++ {
  54. m := t.Method(i)
  55. if strings.HasPrefix(m.Name, "XXX") {
  56. continue
  57. }
  58. if strings.HasPrefix(m.Name, "RegisterModule_") {
  59. id := m.Name[15:]
  60. if _, ok := reflect.TypeOf(this).MethodByName("RegisterModule_" + id); ok {
  61. result := reflect.ValueOf(this).MethodByName("RegisterModule_" + id).Call([]reflect.Value{})
  62. if len(result) >= 1 {
  63. mod := result[0].Interface().(*Module)
  64. mod.Info.Id = id
  65. this.mods[mod.Info.Mount] = mod
  66. }
  67. }
  68. }
  69. if strings.HasPrefix(m.Name, "RegisterAction_") {
  70. id := m.Name[15:]
  71. if _, ok := reflect.TypeOf(this).MethodByName("RegisterAction_" + id); ok {
  72. result := reflect.ValueOf(this).MethodByName("RegisterAction_" + id).Call([]reflect.Value{})
  73. if len(result) >= 1 {
  74. act := result[0].Interface().(*Action)
  75. act.Info.Id = id
  76. this.acts[act.Info.Mount] = act
  77. }
  78. }
  79. }
  80. }
  81. }
  82. func (this *Modules) newModule(info MInfo, ff func(wrap *wrapper.Wrapper), bf func(wrap *wrapper.Wrapper) (string, string, string)) *Module {
  83. return &Module{Info: info, Front: ff, Back: bf}
  84. }
  85. func (this *Modules) newAction(info AInfo, af func(wrap *wrapper.Wrapper)) *Action {
  86. return &Action{Info: info, Act: af}
  87. }
  88. func (this *Modules) getCurrentModule(wrap *wrapper.Wrapper, backend bool) (*Module, string) {
  89. var mod *Module = nil
  90. var modCurr string = ""
  91. // Some module
  92. if len(wrap.UrlArgs) >= 1 {
  93. if m, ok := this.mods[wrap.UrlArgs[0]]; ok {
  94. if (!backend && m.Front != nil) || (backend && m.Back != nil) {
  95. mod = m
  96. modCurr = wrap.UrlArgs[0]
  97. }
  98. }
  99. }
  100. // Default module
  101. if !backend || (backend && len(wrap.UrlArgs) <= 0) {
  102. if mod == nil {
  103. if m, ok := this.mods["index"]; ok {
  104. mod = m
  105. modCurr = "index"
  106. }
  107. }
  108. }
  109. // Selected module
  110. if !backend {
  111. if len(wrap.UrlArgs) <= 0 {
  112. if (*wrap.Config).Engine.MainModule > 0 {
  113. if (*wrap.Config).Engine.MainModule == 1 {
  114. if m, ok := this.mods["blog"]; ok {
  115. mod = m
  116. modCurr = "blog"
  117. }
  118. } else if (*wrap.Config).Engine.MainModule == 2 {
  119. if m, ok := this.mods["shop"]; ok {
  120. mod = m
  121. modCurr = "shop"
  122. }
  123. }
  124. }
  125. }
  126. }
  127. return mod, modCurr
  128. }
  129. func (this *Modules) getModulesList(wrap *wrapper.Wrapper, sys bool, all bool) []*MInfo {
  130. list := make([]*MInfo, 0)
  131. for _, mod := range this.mods {
  132. if mod.Back != nil {
  133. if mod.Info.System == sys || all {
  134. list = append(list, &mod.Info)
  135. }
  136. }
  137. }
  138. sort.Slice(list, func(i, j int) bool {
  139. return list[i].Order < list[j].Order
  140. })
  141. return list
  142. }
  143. func (this *Modules) getSidebarModuleSubMenu(wrap *wrapper.Wrapper, mod *MInfo) string {
  144. html := ``
  145. if mod.Sub != nil {
  146. for _, item := range *mod.Sub {
  147. if item.Show {
  148. if !item.Sep {
  149. class := ""
  150. if (item.Mount == "default" && len(wrap.UrlArgs) <= 1) || (len(wrap.UrlArgs) >= 2 && item.Mount == wrap.UrlArgs[1]) || (len(wrap.UrlArgs) >= 2 && item.Mount == "default" && wrap.UrlArgs[1] == "modify") || (len(wrap.UrlArgs) >= 2 && len(strings.Split(item.Mount, "-")) <= 1 && len(strings.Split(wrap.UrlArgs[1], "-")) >= 2 && strings.Split(wrap.UrlArgs[1], "-")[1] == "modify" && strings.Split(item.Mount, "-")[0] == strings.Split(wrap.UrlArgs[1], "-")[0]) {
  151. class = " active"
  152. }
  153. icon := item.Icon
  154. if icon == "" {
  155. icon = assets.SysSvgIconGear
  156. }
  157. href := "/cp/" + mod.Mount + "/" + item.Mount + "/"
  158. if mod.Mount == "index" && item.Mount == "default" {
  159. href = "/cp/"
  160. } else if item.Mount == "default" {
  161. href = "/cp/" + mod.Mount + "/"
  162. }
  163. html += `<li class="nav-item` + class + `"><a class="nav-link" href="` + href + `">` + icon + item.Name + `</a></li>`
  164. } else {
  165. html += `<li class="nav-separator"></li>`
  166. }
  167. }
  168. }
  169. if html != "" {
  170. html = `<ul class="nav flex-column">` + html + `</ul>`
  171. }
  172. }
  173. return html
  174. }
  175. func (this *Modules) getNavMenuModules(wrap *wrapper.Wrapper, sys bool) string {
  176. html := ``
  177. list := this.getModulesList(wrap, sys, false)
  178. for _, mod := range list {
  179. class := ""
  180. if mod.Mount == wrap.CurrModule {
  181. class = " active"
  182. }
  183. href := `/cp/` + mod.Mount + `/`
  184. if mod.Mount == "index" {
  185. href = `/cp/`
  186. }
  187. if !(sys && (mod.Mount == "api" || mod.Mount == "products")) {
  188. html += `<a class="dropdown-item` + class + `" href="` + href + `">` + mod.Name + `</a>`
  189. }
  190. }
  191. return html
  192. }
  193. func (this *Modules) getSidebarModules(wrap *wrapper.Wrapper) string {
  194. html_def := ""
  195. html_sys := ""
  196. list := this.getModulesList(wrap, false, true)
  197. for _, mod := range list {
  198. class := ""
  199. submenu := ""
  200. if mod.Mount == wrap.CurrModule {
  201. class = " active"
  202. submenu = this.getSidebarModuleSubMenu(wrap, mod)
  203. }
  204. icon := mod.Icon
  205. if icon == "" {
  206. icon = assets.SysSvgIconGear
  207. }
  208. href := "/cp/" + mod.Mount + "/"
  209. if mod.Mount == "index" {
  210. href = "/cp/"
  211. }
  212. if !mod.System {
  213. html_def += `<li class="nav-item` + class + `"><a class="nav-link" href="` + href + `">` + icon + mod.Name + `</a>` + submenu + `</li>`
  214. } else {
  215. if !(mod.Mount == "api" || mod.Mount == "products") {
  216. html_sys += `<li class="nav-item` + class + `"><a class="nav-link" href="` + href + `">` + icon + mod.Name + `</a>` + submenu + `</li>`
  217. }
  218. }
  219. }
  220. if html_def != "" {
  221. html_def = `<ul class="nav flex-column">` + html_def + `</ul>`
  222. }
  223. if html_sys != "" {
  224. html_sys = `<ul class="nav flex-column">` + html_sys + `</ul>`
  225. }
  226. if html_def != "" && html_sys != "" {
  227. html_sys = `<div class="dropdown-divider"></div>` + html_sys
  228. }
  229. return html_def + html_sys
  230. }
  231. func (this *Modules) getBreadCrumbs(wrap *wrapper.Wrapper, data *[]consts.BreadCrumb) string {
  232. res := `<nav aria-label="breadcrumb">`
  233. res += `<ol class="breadcrumb">`
  234. if this.mods[wrap.CurrModule].Info.Mount == "index" {
  235. res += `<li class="breadcrumb-item"><a href="/cp/">` + html.EscapeString(this.mods[wrap.CurrModule].Info.Name) + `</a></li>`
  236. } else {
  237. res += `<li class="breadcrumb-item"><a href="/cp/` + this.mods[wrap.CurrModule].Info.Mount + `/">` + html.EscapeString(this.mods[wrap.CurrModule].Info.Name) + `</a></li>`
  238. }
  239. for _, item := range *data {
  240. if item.Link == "" {
  241. res += `<li class="breadcrumb-item active" aria-current="page">` + html.EscapeString(item.Name) + `</li>`
  242. } else {
  243. res += `<li class="breadcrumb-item"><a href="` + item.Link + `">` + html.EscapeString(item.Name) + `</a></li>`
  244. }
  245. }
  246. res += `</ol>`
  247. res += `</nav>`
  248. return res
  249. }
  250. func New() *Modules {
  251. m := Modules{
  252. mods: map[string]*Module{},
  253. acts: map[string]*Action{},
  254. }
  255. m.load()
  256. return &m
  257. }
  258. func (this *Modules) XXXActionHeaders(wrap *wrapper.Wrapper, status int) {
  259. wrap.W.WriteHeader(status)
  260. wrap.W.Header().Set("Cache-Control", "no-cache, no-store, must-revalidate")
  261. wrap.W.Header().Set("Content-Type", "text/html; charset=utf-8")
  262. }
  263. func (this *Modules) XXXActionFire(wrap *wrapper.Wrapper) bool {
  264. if wrap.R.Method == "POST" {
  265. if err := wrap.R.ParseForm(); err == nil {
  266. name := wrap.R.FormValue("action")
  267. if name == "" {
  268. wrap.R.ParseMultipartForm(32 << 20)
  269. name = wrap.R.FormValue("action")
  270. }
  271. if name != "" {
  272. if act, ok := this.acts[name]; ok {
  273. if act.Info.WantDB {
  274. err := wrap.UseDatabase()
  275. if err != nil {
  276. this.XXXActionHeaders(wrap, http.StatusNotFound)
  277. wrap.MsgError(err.Error())
  278. return true
  279. }
  280. }
  281. if act.Info.WantUser || act.Info.WantAdmin {
  282. if !wrap.LoadSessionUser() {
  283. this.XXXActionHeaders(wrap, http.StatusNotFound)
  284. wrap.MsgError(`You must be loginned to run this action`)
  285. return true
  286. }
  287. if wrap.User.A_active <= 0 {
  288. if !wrap.LoadSessionUser() {
  289. this.XXXActionHeaders(wrap, http.StatusNotFound)
  290. wrap.MsgError(`You do not have rights to run this action`)
  291. return true
  292. }
  293. }
  294. }
  295. if act.Info.WantAdmin && wrap.User.A_admin <= 0 {
  296. if !wrap.LoadSessionUser() {
  297. this.XXXActionHeaders(wrap, http.StatusNotFound)
  298. wrap.MsgError(`You do not have rights to run this action`)
  299. return true
  300. }
  301. }
  302. this.XXXActionHeaders(wrap, http.StatusOK)
  303. act.Act(wrap)
  304. return true
  305. } else {
  306. this.XXXActionHeaders(wrap, http.StatusNotFound)
  307. wrap.MsgError(`This action is not implemented`)
  308. return true
  309. }
  310. }
  311. }
  312. }
  313. return false
  314. }
  315. func (this *Modules) XXXFrontEnd(wrap *wrapper.Wrapper) bool {
  316. mod, cm := this.getCurrentModule(wrap, false)
  317. if mod != nil {
  318. wrap.CurrModule = cm
  319. if mod.Front != nil {
  320. if mod.Info.WantDB {
  321. err := wrap.UseDatabase()
  322. if err != nil {
  323. utils.SystemErrorPageEngine(wrap.W, err)
  324. return true
  325. }
  326. }
  327. mod.Front(wrap)
  328. return true
  329. }
  330. }
  331. return false
  332. }
  333. func (this *Modules) XXXBackEnd(wrap *wrapper.Wrapper) bool {
  334. mod, cm := this.getCurrentModule(wrap, true)
  335. if mod != nil {
  336. wrap.CurrModule = cm
  337. if len(wrap.UrlArgs) >= 2 && wrap.UrlArgs[1] != "" {
  338. wrap.CurrSubModule = wrap.UrlArgs[1]
  339. }
  340. // Search for sub module mount
  341. found := false
  342. submount := "default"
  343. if wrap.CurrSubModule != "" {
  344. submount = wrap.CurrSubModule
  345. }
  346. for _, item := range *mod.Info.Sub {
  347. if item.Mount == submount {
  348. found = true
  349. break
  350. }
  351. }
  352. // Display standart 404 error page
  353. if !found {
  354. return found
  355. }
  356. // Call module function
  357. if mod.Back != nil {
  358. sidebar_left, content, sidebar_right := mod.Back(wrap)
  359. // Display standart 404 error page
  360. if sidebar_left == "" && content == "" && sidebar_right == "" {
  361. return false
  362. }
  363. // Prepare CP page
  364. body_class := "cp"
  365. if sidebar_left != "" {
  366. body_class = body_class + " cp-sidebar-left"
  367. }
  368. if content == "" {
  369. body_class = body_class + " cp-404"
  370. }
  371. if sidebar_right != "" {
  372. body_class = body_class + " cp-sidebar-right"
  373. }
  374. wrap.RenderBackEnd(assets.TmplCpBase, consts.TmplDataCpBase{
  375. Title: wrap.CurrHost + " - Fave " + consts.ServerVersion,
  376. Caption: "Fave " + consts.ServerVersion,
  377. BodyClasses: body_class,
  378. UserId: wrap.User.A_id,
  379. UserFirstName: utils.JavaScriptVarValue(wrap.User.A_first_name),
  380. UserLastName: utils.JavaScriptVarValue(wrap.User.A_last_name),
  381. UserEmail: utils.JavaScriptVarValue(wrap.User.A_email),
  382. UserAvatarLink: "https://s.gravatar.com/avatar/" + utils.GetMd5(wrap.User.A_email) + "?s=80&r=g",
  383. NavBarModules: template.HTML(this.getNavMenuModules(wrap, false)),
  384. NavBarModulesSys: template.HTML(this.getNavMenuModules(wrap, true)),
  385. ModuleCurrentAlias: wrap.CurrModule,
  386. SidebarLeft: template.HTML(sidebar_left),
  387. Content: template.HTML(content),
  388. SidebarRight: template.HTML(sidebar_right),
  389. })
  390. return true
  391. }
  392. }
  393. return false
  394. }