9b1489b90f
Each theme has more of a distinct feel in the control panel now. I moved a large portion of the inline CSS into global.css, tweaked some core code which didn't line up, and added CSS text insertions to make things more flexible. Revamped the alerts interface. We have more changes coming up! Added screenshots for Tempra Cursive and Tempra Conflux Mobile to the README. Added a .htaccess file, just in case someone plops Gosora in Apache's /www/ folder to stop the contents of config.go from becoming publically visible. Never put Gosora in your /www/ folder, Gosora is a standalone program which does NOT use Apache or any other webserver. Fixed a bug in the inline forum editor. Added a hand-written Markdown parser which is much faster than the previous Regex based one. This is still a work in progress. Added support for strikethrough and underline elements to the Markdown parser. Fixed the missing semicolons in global.js Revamped the form CSS. Author bits on the theme manager now link to the author's website. Improved the profiles a little. The code in the stylesheets now have a more consistent style. Fixed a bug in the Cosmo theme relating to the fact that Gosora doesn't have sidebars yet. There are many more changes which aren't listed here. If weirdness regarding lines or spacing occurs, I'm experimenting with a new text editor. I hope to have this fixed by the next commit.
277 lines
7.7 KiB
Go
277 lines
7.7 KiB
Go
/* Copyright Azareal 2016 - 2018 */
|
|
package main
|
|
|
|
import (
|
|
//"fmt"
|
|
"log"
|
|
"io"
|
|
"os"
|
|
"strings"
|
|
"mime"
|
|
"io/ioutil"
|
|
"path/filepath"
|
|
"encoding/json"
|
|
"net/http"
|
|
)
|
|
|
|
var defaultTheme string
|
|
var themes map[string]Theme = make(map[string]Theme)
|
|
//var overriden_templates map[string]interface{} = make(map[string]interface{})
|
|
var overriden_templates map[string]bool = make(map[string]bool)
|
|
|
|
type Theme struct
|
|
{
|
|
Name string
|
|
FriendlyName string
|
|
Version string
|
|
Creator string
|
|
FullImage string
|
|
MobileFriendly bool
|
|
Disabled bool
|
|
HideFromThemes bool
|
|
ForkOf string
|
|
Tag string
|
|
URL string
|
|
Settings map[string]ThemeSetting
|
|
Templates []TemplateMapping
|
|
|
|
// This variable should only be set and unset by the system, not the theme meta file
|
|
Active bool
|
|
}
|
|
|
|
type ThemeSetting struct
|
|
{
|
|
FriendlyName string
|
|
Options []string
|
|
}
|
|
|
|
type TemplateMapping struct
|
|
{
|
|
Name string
|
|
Source string
|
|
//When string
|
|
}
|
|
|
|
func init_themes() {
|
|
themeFiles, err := ioutil.ReadDir("./themes")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
for _, themeFile := range themeFiles {
|
|
if !themeFile.IsDir() {
|
|
continue
|
|
}
|
|
|
|
themeName := themeFile.Name()
|
|
log.Print("Adding theme '" + themeName + "'")
|
|
themeFile, err := ioutil.ReadFile("./themes/" + themeName + "/theme.json")
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
var theme Theme
|
|
err = json.Unmarshal(themeFile, &theme)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
|
|
theme.Active = false // Set this to false, just in case someone explicitly overrode this value in the JSON file
|
|
|
|
if theme.FullImage != "" {
|
|
if debug {
|
|
log.Print("Adding theme image")
|
|
}
|
|
err = add_static_file("./themes/" + themeName + "/" + theme.FullImage, "./themes/" + themeName)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}
|
|
|
|
themes[theme.Name] = theme
|
|
}
|
|
}
|
|
|
|
func add_theme_static_files(themeName string) {
|
|
err := filepath.Walk("./themes/" + themeName + "/public", func(path string, f os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if f.IsDir() {
|
|
return nil
|
|
}
|
|
path = strings.Replace(path,"\\","/",-1)
|
|
|
|
if debug {
|
|
log.Print("Attempting to add static file '" + path + "' for default theme '" + themeName + "'")
|
|
}
|
|
|
|
data, err := ioutil.ReadFile(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
path = strings.TrimPrefix(path,"themes/" + themeName + "/public")
|
|
if debug {
|
|
log.Print("Added the '" + path + "' static file for default theme " + themeName + ".")
|
|
}
|
|
gzip_data := compress_bytes_gzip(data)
|
|
|
|
static_files["/static" + path] = SFile{data,gzip_data,0,int64(len(data)),int64(len(gzip_data)),mime.TypeByExtension(filepath.Ext("/themes/" + themeName + "/public" + path)),f,f.ModTime().UTC().Format(http.TimeFormat)}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
}
|
|
|
|
func map_theme_templates(theme Theme) {
|
|
if theme.Templates != nil {
|
|
for _, themeTmpl := range theme.Templates {
|
|
if themeTmpl.Name == "" {
|
|
log.Fatal("Invalid destination template name")
|
|
}
|
|
if themeTmpl.Source == "" {
|
|
log.Fatal("Invalid source template name")
|
|
}
|
|
|
|
// `go generate` is one possibility for letting plugins inject custom page structs, but it would simply add another step of compilation. It might be simpler than the current build process from the perspective of the administrator?
|
|
|
|
dest_tmpl_ptr, ok := tmpl_ptr_map[themeTmpl.Name]
|
|
if !ok {
|
|
log.Fatal("The destination template doesn't exist!")
|
|
}
|
|
source_tmpl_ptr, ok := tmpl_ptr_map[themeTmpl.Source]
|
|
if !ok {
|
|
log.Fatal("The source template doesn't exist!")
|
|
}
|
|
|
|
switch d_tmpl_ptr := dest_tmpl_ptr.(type) {
|
|
case *func(TopicPage,io.Writer):
|
|
switch s_tmpl_ptr := source_tmpl_ptr.(type) {
|
|
case *func(TopicPage,io.Writer):
|
|
//overriden_templates[themeTmpl.Name] = d_tmpl_ptr
|
|
overriden_templates[themeTmpl.Name] = true
|
|
*d_tmpl_ptr = *s_tmpl_ptr
|
|
default:
|
|
log.Fatal("The source and destination templates are incompatible")
|
|
}
|
|
case *func(TopicsPage,io.Writer):
|
|
switch s_tmpl_ptr := source_tmpl_ptr.(type) {
|
|
case *func(TopicsPage,io.Writer):
|
|
//overriden_templates[themeTmpl.Name] = d_tmpl_ptr
|
|
overriden_templates[themeTmpl.Name] = true
|
|
*d_tmpl_ptr = *s_tmpl_ptr
|
|
default:
|
|
log.Fatal("The source and destination templates are incompatible")
|
|
}
|
|
case *func(ForumPage,io.Writer):
|
|
switch s_tmpl_ptr := source_tmpl_ptr.(type) {
|
|
case *func(ForumPage,io.Writer):
|
|
//overriden_templates[themeTmpl.Name] = d_tmpl_ptr
|
|
overriden_templates[themeTmpl.Name] = true
|
|
*d_tmpl_ptr = *s_tmpl_ptr
|
|
default:
|
|
log.Fatal("The source and destination templates are incompatible")
|
|
}
|
|
case *func(ForumsPage,io.Writer):
|
|
switch s_tmpl_ptr := source_tmpl_ptr.(type) {
|
|
case *func(ForumsPage,io.Writer):
|
|
//overriden_templates[themeTmpl.Name] = d_tmpl_ptr
|
|
overriden_templates[themeTmpl.Name] = true
|
|
*d_tmpl_ptr = *s_tmpl_ptr
|
|
default:
|
|
log.Fatal("The source and destination templates are incompatible")
|
|
}
|
|
case *func(ProfilePage,io.Writer):
|
|
switch s_tmpl_ptr := source_tmpl_ptr.(type) {
|
|
case *func(ProfilePage,io.Writer):
|
|
//overriden_templates[themeTmpl.Name] = d_tmpl_ptr
|
|
overriden_templates[themeTmpl.Name] = true
|
|
*d_tmpl_ptr = *s_tmpl_ptr
|
|
default:
|
|
log.Fatal("The source and destination templates are incompatible")
|
|
}
|
|
case *func(Page,io.Writer):
|
|
switch s_tmpl_ptr := source_tmpl_ptr.(type) {
|
|
case *func(Page,io.Writer):
|
|
//overriden_templates[themeTmpl.Name] = d_tmpl_ptr
|
|
overriden_templates[themeTmpl.Name] = true
|
|
*d_tmpl_ptr = *s_tmpl_ptr
|
|
default:
|
|
log.Fatal("The source and destination templates are incompatible")
|
|
}
|
|
default:
|
|
log.Fatal("Unknown destination template type!")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func reset_template_overrides() {
|
|
log.Print("Resetting the template overrides")
|
|
|
|
for name, _ := range overriden_templates {
|
|
log.Print("Resetting '" + name + "' template override")
|
|
|
|
origin_pointer, ok := tmpl_ptr_map["o_" + name]
|
|
if !ok {
|
|
//log.Fatal("The origin template doesn't exist!")
|
|
log.Print("The origin template doesn't exist!")
|
|
return
|
|
}
|
|
|
|
dest_tmpl_ptr, ok := tmpl_ptr_map[name]
|
|
if !ok {
|
|
//log.Fatal("The destination template doesn't exist!")
|
|
log.Print("The destination template doesn't exist!")
|
|
return
|
|
}
|
|
|
|
// Not really a pointer, more of a function handle, an artifact from one of the earlier versions of themes.go
|
|
switch o_ptr := origin_pointer.(type) {
|
|
case func(TopicPage,io.Writer):
|
|
switch d_ptr := dest_tmpl_ptr.(type) {
|
|
case *func(TopicPage,io.Writer):
|
|
*d_ptr = o_ptr
|
|
default:
|
|
log.Fatal("The origin and destination templates are incompatible")
|
|
}
|
|
case func(TopicsPage,io.Writer):
|
|
switch d_ptr := dest_tmpl_ptr.(type) {
|
|
case *func(TopicsPage,io.Writer):
|
|
*d_ptr = o_ptr
|
|
default:
|
|
log.Fatal("The origin and destination templates are incompatible")
|
|
}
|
|
case func(ForumPage,io.Writer):
|
|
switch d_ptr := dest_tmpl_ptr.(type) {
|
|
case *func(ForumPage,io.Writer):
|
|
*d_ptr = o_ptr
|
|
default:
|
|
log.Fatal("The origin and destination templates are incompatible")
|
|
}
|
|
case func(ForumsPage,io.Writer):
|
|
switch d_ptr := dest_tmpl_ptr.(type) {
|
|
case *func(ForumsPage,io.Writer):
|
|
*d_ptr = o_ptr
|
|
default:
|
|
log.Fatal("The origin and destination templates are incompatible")
|
|
}
|
|
case func(ProfilePage,io.Writer):
|
|
switch d_ptr := dest_tmpl_ptr.(type) {
|
|
case *func(ProfilePage,io.Writer):
|
|
*d_ptr = o_ptr
|
|
default:
|
|
log.Fatal("The origin and destination templates are incompatible")
|
|
}
|
|
default:
|
|
log.Fatal("Unknown destination template type!")
|
|
}
|
|
log.Print("The template override was reset")
|
|
}
|
|
overriden_templates = make(map[string]bool)
|
|
log.Print("All of the template overrides have been reset")
|
|
}
|