data_table.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. package builder
  2. import (
  3. "database/sql"
  4. "fmt"
  5. "html"
  6. "math"
  7. "strconv"
  8. "golang-fave/engine/wrapper"
  9. )
  10. type DataTableRow struct {
  11. DBField string
  12. DBExp string
  13. NameInTable string
  14. Classes string
  15. CallBack func(values *[]string) string
  16. }
  17. func DataTable(
  18. wrap *wrapper.Wrapper,
  19. table string,
  20. order_by string,
  21. order_way string,
  22. data *[]DataTableRow,
  23. action func(values *[]string) string,
  24. pagination_url string,
  25. custom_sql_count func() (int, error),
  26. custom_sql_data func(limit_offset int, pear_page int) (*sql.Rows, error),
  27. ) string {
  28. var num int
  29. var err error
  30. if custom_sql_count != nil {
  31. num, err = custom_sql_count()
  32. } else {
  33. err = wrap.DB.QueryRow("SELECT COUNT(*) FROM `" + table + "`;").Scan(&num)
  34. if err != nil {
  35. return ""
  36. }
  37. }
  38. pear_page := 10
  39. max_pages := int(math.Ceil(float64(num) / float64(pear_page)))
  40. curr_page := 1
  41. p := wrap.R.URL.Query().Get("p")
  42. if p != "" {
  43. pi, err := strconv.Atoi(p)
  44. if err != nil {
  45. curr_page = 1
  46. } else {
  47. if pi < 1 {
  48. curr_page = 1
  49. } else if pi > max_pages {
  50. curr_page = max_pages
  51. } else {
  52. curr_page = pi
  53. }
  54. }
  55. }
  56. limit_offset := curr_page*pear_page - pear_page
  57. result := `<table id="cp-table-` + table + `" class="table data-table table-striped table-bordered table-hover table_` + table + `">`
  58. result += `<thead>`
  59. result += `<tr>`
  60. qsql := ""
  61. if custom_sql_data == nil {
  62. qsql = "SELECT"
  63. }
  64. for i, column := range *data {
  65. if column.NameInTable != "" {
  66. classes := column.Classes
  67. if classes != "" {
  68. classes = " " + classes
  69. }
  70. result += `<th scope="col" class="col_` + column.DBField + classes + `">` + html.EscapeString(column.NameInTable) + `</th>`
  71. }
  72. if custom_sql_data == nil {
  73. if column.DBExp == "" {
  74. qsql += " `" + column.DBField + "`"
  75. } else {
  76. qsql += " " + column.DBExp + " as `" + column.DBField + "`"
  77. }
  78. if i+1 < len(*data) {
  79. qsql += ","
  80. }
  81. }
  82. }
  83. if custom_sql_data == nil {
  84. qsql += " FROM `" + table + "` ORDER BY `" + order_by + "` " + order_way + " LIMIT ?, ?;"
  85. }
  86. if action != nil {
  87. result += `<th scope="col" class="col_action">&nbsp;</th>`
  88. }
  89. result += `</tr>`
  90. result += `</thead>`
  91. result += `<tbody>`
  92. if num > 0 {
  93. var rows *sql.Rows
  94. var err error
  95. if custom_sql_data == nil {
  96. rows, err = wrap.DB.Query(qsql, limit_offset, pear_page)
  97. } else {
  98. rows, err = custom_sql_data(limit_offset, pear_page)
  99. }
  100. if err == nil {
  101. values := make([]string, len(*data))
  102. scan := make([]interface{}, len(values))
  103. for i := range values {
  104. scan[i] = &values[i]
  105. }
  106. for rows.Next() {
  107. err = rows.Scan(scan...)
  108. if err == nil {
  109. result += `<tr>`
  110. for i, val := range values {
  111. if (*data)[i].NameInTable != "" {
  112. classes := (*data)[i].Classes
  113. if classes != "" {
  114. classes = " " + classes
  115. }
  116. if (*data)[i].CallBack == nil {
  117. result += `<td class="col_` + (*data)[i].DBField + classes + `">` + html.EscapeString(string(val)) + `</td>`
  118. } else {
  119. result += `<td class="col_` + (*data)[i].DBField + classes + `">` + (*data)[i].CallBack(&values) + `</td>`
  120. }
  121. }
  122. }
  123. if action != nil {
  124. result += `<td class="col_action">` + action(&values) + `</td>`
  125. }
  126. result += `</tr>`
  127. }
  128. }
  129. }
  130. } else {
  131. result += `<tr><td colspan="50">No any data found</td></tr>`
  132. }
  133. result += `</tbody></table>`
  134. // Show page navigation only if pages more then one
  135. if max_pages > 1 {
  136. result += `<nav>`
  137. result += `<ul class="pagination" style="margin-bottom:0px;">`
  138. class := ""
  139. if curr_page <= 1 {
  140. class = " disabled"
  141. }
  142. result += `<li class="page-item` + class + `">`
  143. result += `<a class="page-link" href="` + pagination_url + `?p=` + fmt.Sprintf("%d", curr_page-1) + `" aria-label="Previous">`
  144. result += `<span aria-hidden="true">&laquo;</span>`
  145. result += `<span class="sr-only">Previous</span>`
  146. result += `</a>`
  147. result += `</li>`
  148. for i := 1; i <= max_pages; i++ {
  149. class = ""
  150. if i == curr_page {
  151. class = " active"
  152. }
  153. result += `<li class="page-item` + class + `">`
  154. result += `<a class="page-link" href="` + pagination_url + `?p=` + fmt.Sprintf("%d", i) + `">` + fmt.Sprintf("%d", i) + `</a>`
  155. result += `</li>`
  156. }
  157. class = ""
  158. if curr_page >= max_pages {
  159. class = " disabled"
  160. }
  161. result += `<li class="page-item` + class + `">`
  162. result += `<a class="page-link" href="` + pagination_url + `?p=` + fmt.Sprintf("%d", curr_page+1) + `" aria-label="Next">`
  163. result += `<span aria-hidden="true">&raquo;</span>`
  164. result += `<span class="sr-only">Next</span>`
  165. result += `</a>`
  166. result += `</li>`
  167. result += `</ul>`
  168. result += `</nav>`
  169. }
  170. return result
  171. }