data_table.go 5.0 KB

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