Overhauled the widget system. You can now specify more complex logic for where a widget can show up and you can now place widgets in the footer.
Added fsnotify as a dependency, we'll be using it, Rez, and Riot (maybe) soon. Removed the intercept_build_widgets vhook. You can now pass an ID to BlankReply() for mocking replies or whatever else you want to do with this. Added the dock template function for widget docks. Added the logf internal function to the template transpiler and did some minor clean-up. Removed the Sidebars property from theme.json and added the more general Docks one. Improved the footer and the associated CSS. Added the about widget, previously the AboutSegment feature. Removed the about_segment_title and about_segment_body settings. Gosora now exits when it receives the appropriate OS signal. Refactored the modlogs route. More progress on the theme in the Control Panel.
This commit is contained in:
parent
381ce3083a
commit
0e9cebfa47
|
@ -111,6 +111,8 @@ go get -u github.com/lib/pq
|
||||||
|
|
||||||
go get -u github.com/denisenkom/go-mssqldb
|
go get -u github.com/denisenkom/go-mssqldb
|
||||||
|
|
||||||
|
go get -u github.com/fsnotify/fsnotify
|
||||||
|
|
||||||
|
|
||||||
go generate
|
go generate
|
||||||
|
|
||||||
|
@ -166,7 +168,7 @@ We're looking for ways to clean-up the plugin system so that all of them (except
|
||||||
|
|
||||||
More images in the /images/ folder. Beware though, some of them are *really* outdated.
|
More images in the /images/ folder. Beware though, some of them are *really* outdated.
|
||||||
|
|
||||||
# Dependencies
|
# Dependencies (a few of these like Riot aren't currently in use, but we anticipate that we'll need some sort of search engine library in the very immediate future)
|
||||||
|
|
||||||
* Go 1.9
|
* Go 1.9
|
||||||
|
|
||||||
|
@ -194,6 +196,10 @@ More images in the /images/ folder. Beware though, some of them are *really* out
|
||||||
|
|
||||||
* github.com/go-ego/riot A search engine library.
|
* github.com/go-ego/riot A search engine library.
|
||||||
|
|
||||||
|
* github.com/bamiaux/rez An image resizer (e.g. for spitting out thumbnails)
|
||||||
|
|
||||||
|
* github.com/fsnotify/fsnotify A library for watching events on the file system.
|
||||||
|
|
||||||
# Bundled Plugins
|
# Bundled Plugins
|
||||||
|
|
||||||
There are several plugins which are bundled with the software by default. These cover various common tasks which aren't common enough to clutter the core with or which have competing implementation methods (E.g. plugin_markdown vs plugin_bbcode for post mark-up).
|
There are several plugins which are bundled with the software by default. These cover various common tasks which aren't common enough to clutter the core with or which have competing implementation methods (E.g. plugin_markdown vs plugin_bbcode for post mark-up).
|
||||||
|
|
|
@ -26,7 +26,6 @@ var Hooks = map[string][]func(interface{}) interface{}{
|
||||||
|
|
||||||
// Hooks with a variable number of arguments
|
// Hooks with a variable number of arguments
|
||||||
var Vhooks = map[string]func(...interface{}) interface{}{
|
var Vhooks = map[string]func(...interface{}) interface{}{
|
||||||
"intercept_build_widgets": nil,
|
|
||||||
"forum_trow_assign": nil,
|
"forum_trow_assign": nil,
|
||||||
"topics_topic_row_assign": nil,
|
"topics_topic_row_assign": nil,
|
||||||
//"topics_user_row_assign": nil,
|
//"topics_user_row_assign": nil,
|
||||||
|
|
|
@ -2,6 +2,7 @@ package common
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"html/template"
|
"html/template"
|
||||||
|
"net/http"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
@ -16,6 +17,8 @@ type HeaderVars struct {
|
||||||
Themes map[string]Theme // TODO: Use a slice containing every theme instead of the main map for speed?
|
Themes map[string]Theme // TODO: Use a slice containing every theme instead of the main map for speed?
|
||||||
Theme Theme
|
Theme Theme
|
||||||
//TemplateName string // TODO: Use this to move template calls to the router rather than duplicating them over and over and over?
|
//TemplateName string // TODO: Use this to move template calls to the router rather than duplicating them over and over and over?
|
||||||
|
Zone string
|
||||||
|
Writer http.ResponseWriter
|
||||||
ExtData ExtData
|
ExtData ExtData
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -131,6 +131,10 @@ func (reply *Reply) Copy() Reply {
|
||||||
return *reply
|
return *reply
|
||||||
}
|
}
|
||||||
|
|
||||||
func BlankReply() *Reply {
|
func BlankReply(ids ...int) *Reply {
|
||||||
return &Reply{ID: 0}
|
var id int
|
||||||
|
if len(ids) != 0 {
|
||||||
|
id = ids[0]
|
||||||
|
}
|
||||||
|
return &Reply{ID: id}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,7 +2,6 @@ package common
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"html"
|
"html"
|
||||||
"html/template"
|
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
|
@ -22,30 +21,6 @@ var MemberCheck func(w http.ResponseWriter, r *http.Request, user *User) (header
|
||||||
var SimpleUserCheck func(w http.ResponseWriter, r *http.Request, user *User) (headerLite *HeaderLite, err RouteError) = simpleUserCheck
|
var SimpleUserCheck func(w http.ResponseWriter, r *http.Request, user *User) (headerLite *HeaderLite, err RouteError) = simpleUserCheck
|
||||||
var UserCheck func(w http.ResponseWriter, r *http.Request, user *User) (headerVars *HeaderVars, err RouteError) = userCheck
|
var UserCheck func(w http.ResponseWriter, r *http.Request, user *User) (headerVars *HeaderVars, err RouteError) = userCheck
|
||||||
|
|
||||||
// TODO: Support for left sidebars and sidebars on both sides
|
|
||||||
// http.Request is for context.Context middleware. Mostly for plugin_guilds right now
|
|
||||||
func BuildWidgets(zone string, data interface{}, headerVars *HeaderVars, r *http.Request) {
|
|
||||||
if Vhooks["intercept_build_widgets"] != nil {
|
|
||||||
if RunVhook("intercept_build_widgets", zone, data, headerVars, r).(bool) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if Themes[headerVars.Theme.Name].Sidebars == "right" {
|
|
||||||
if len(Docks.RightSidebar) != 0 {
|
|
||||||
var sbody string
|
|
||||||
for _, widget := range Docks.RightSidebar {
|
|
||||||
if widget.Enabled {
|
|
||||||
if widget.Location == "global" || widget.Location == zone {
|
|
||||||
sbody += widget.Body
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
headerVars.Widgets.RightSidebar = template.HTML(sbody)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func simpleForumUserCheck(w http.ResponseWriter, r *http.Request, user *User, fid int) (headerLite *HeaderLite, rerr RouteError) {
|
func simpleForumUserCheck(w http.ResponseWriter, r *http.Request, user *User, fid int) (headerLite *HeaderLite, rerr RouteError) {
|
||||||
if !Forums.Exists(fid) {
|
if !Forums.Exists(fid) {
|
||||||
return nil, PreError("The target forum doesn't exist.", w, r)
|
return nil, PreError("The target forum doesn't exist.", w, r)
|
||||||
|
@ -140,6 +115,7 @@ func panelUserCheck(w http.ResponseWriter, r *http.Request, user *User) (headerV
|
||||||
Settings: SettingBox.Load().(SettingMap),
|
Settings: SettingBox.Load().(SettingMap),
|
||||||
Themes: Themes,
|
Themes: Themes,
|
||||||
Theme: theme,
|
Theme: theme,
|
||||||
|
Zone: "panel",
|
||||||
}
|
}
|
||||||
// TODO: We should probably initialise headerVars.ExtData
|
// TODO: We should probably initialise headerVars.ExtData
|
||||||
|
|
||||||
|
@ -231,6 +207,7 @@ func userCheck(w http.ResponseWriter, r *http.Request, user *User) (headerVars *
|
||||||
Settings: SettingBox.Load().(SettingMap),
|
Settings: SettingBox.Load().(SettingMap),
|
||||||
Themes: Themes,
|
Themes: Themes,
|
||||||
Theme: theme,
|
Theme: theme,
|
||||||
|
Zone: "frontend",
|
||||||
}
|
}
|
||||||
|
|
||||||
if user.IsBanned {
|
if user.IsBanned {
|
||||||
|
|
|
@ -138,7 +138,6 @@ func compileTemplates() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, forum := range forums {
|
for _, forum := range forums {
|
||||||
//log.Printf("*forum %+v\n", *forum)
|
|
||||||
forumList = append(forumList, *forum)
|
forumList = append(forumList, *forum)
|
||||||
}
|
}
|
||||||
varList = make(map[string]tmpl.VarItem)
|
varList = make(map[string]tmpl.VarItem)
|
||||||
|
@ -272,6 +271,10 @@ func InitTemplates() error {
|
||||||
return leftInt / rightInt
|
return leftInt / rightInt
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fmap["dock"] = func(dock interface{}, headerVarInt interface{}) interface{} {
|
||||||
|
return template.HTML(BuildWidget(dock.(string), headerVarInt.(*HeaderVars)))
|
||||||
|
}
|
||||||
|
|
||||||
// The interpreted templates...
|
// The interpreted templates...
|
||||||
if Dev.DebugMode {
|
if Dev.DebugMode {
|
||||||
log.Print("Loading the template files...")
|
log.Print("Loading the template files...")
|
||||||
|
|
|
@ -91,6 +91,7 @@ func (c *CTemplateSet) Compile(name string, dir string, expects string, expectsI
|
||||||
"subtract": true,
|
"subtract": true,
|
||||||
"multiply": true,
|
"multiply": true,
|
||||||
"divide": true,
|
"divide": true,
|
||||||
|
"dock": true,
|
||||||
}
|
}
|
||||||
|
|
||||||
c.importMap = map[string]string{
|
c.importMap = map[string]string{
|
||||||
|
@ -450,7 +451,10 @@ func (c *CTemplateSet) compileSubswitch(varholder string, holdreflect reflect.Va
|
||||||
case *parse.IdentifierNode:
|
case *parse.IdentifierNode:
|
||||||
c.log("Identifier Node:", node)
|
c.log("Identifier Node:", node)
|
||||||
c.log("Identifier Node Args:", node.Args)
|
c.log("Identifier Node Args:", node.Args)
|
||||||
out, outval := c.compileIdentSwitch(varholder, holdreflect, templateName, node)
|
out, outval, lit := c.compileIdentSwitch(varholder, holdreflect, templateName, node)
|
||||||
|
if lit {
|
||||||
|
return out
|
||||||
|
}
|
||||||
return c.compileVarsub(out, outval, "")
|
return c.compileVarsub(out, outval, "")
|
||||||
default:
|
default:
|
||||||
return c.unknownNode(node)
|
return c.unknownNode(node)
|
||||||
|
@ -507,7 +511,7 @@ func (c *CTemplateSet) unknownNode(node parse.Node) (out string) {
|
||||||
|
|
||||||
func (c *CTemplateSet) compileIdentSwitchN(varholder string, holdreflect reflect.Value, templateName string, node *parse.CommandNode) (out string) {
|
func (c *CTemplateSet) compileIdentSwitchN(varholder string, holdreflect reflect.Value, templateName string, node *parse.CommandNode) (out string) {
|
||||||
c.log("in compileIdentSwitchN")
|
c.log("in compileIdentSwitchN")
|
||||||
out, _ = c.compileIdentSwitch(varholder, holdreflect, templateName, node)
|
out, _, _ = c.compileIdentSwitch(varholder, holdreflect, templateName, node)
|
||||||
return out
|
return out
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -541,7 +545,7 @@ func (c *CTemplateSet) simpleMath(varholder string, holdreflect reflect.Value, t
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *CTemplateSet) compareJoin(varholder string, holdreflect reflect.Value, templateName string, pos int, node *parse.CommandNode, symbol string) (pos2 int, out string) {
|
func (c *CTemplateSet) compareJoin(varholder string, holdreflect reflect.Value, templateName string, pos int, node *parse.CommandNode, symbol string) (pos2 int, out string) {
|
||||||
c.log("Building " + symbol + " function")
|
c.logf("Building %s function", symbol)
|
||||||
if pos == 0 {
|
if pos == 0 {
|
||||||
fmt.Println("pos:", pos)
|
fmt.Println("pos:", pos)
|
||||||
panic(symbol + " is missing a left operand")
|
panic(symbol + " is missing a left operand")
|
||||||
|
@ -572,7 +576,7 @@ func (c *CTemplateSet) compareJoin(varholder string, holdreflect reflect.Value,
|
||||||
return pos, out
|
return pos, out
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *CTemplateSet) compileIdentSwitch(varholder string, holdreflect reflect.Value, templateName string, node *parse.CommandNode) (out string, val reflect.Value) {
|
func (c *CTemplateSet) compileIdentSwitch(varholder string, holdreflect reflect.Value, templateName string, node *parse.CommandNode) (out string, val reflect.Value, literal bool) {
|
||||||
c.log("in compileIdentSwitch")
|
c.log("in compileIdentSwitch")
|
||||||
ArgLoop:
|
ArgLoop:
|
||||||
for pos := 0; pos < len(node.Args); pos++ {
|
for pos := 0; pos < len(node.Args); pos++ {
|
||||||
|
@ -624,6 +628,35 @@ ArgLoop:
|
||||||
out += rout
|
out += rout
|
||||||
val = rval
|
val = rval
|
||||||
break ArgLoop
|
break ArgLoop
|
||||||
|
case "dock":
|
||||||
|
var leftParam, rightParam string
|
||||||
|
// TODO: Implement string literals properly
|
||||||
|
leftOperand := node.Args[pos+1].String()
|
||||||
|
rightOperand := node.Args[pos+2].String()
|
||||||
|
|
||||||
|
if len(leftOperand) == 0 || len(rightOperand) == 0 {
|
||||||
|
panic("The left or right operand for function dock cannot be left blank")
|
||||||
|
}
|
||||||
|
if leftOperand[0] == '"' {
|
||||||
|
leftParam = leftOperand
|
||||||
|
} else {
|
||||||
|
leftParam, _ = c.compileIfVarsub(leftOperand, varholder, templateName, holdreflect)
|
||||||
|
}
|
||||||
|
if rightOperand[0] == '"' {
|
||||||
|
panic("The right operand for function dock cannot be a string")
|
||||||
|
}
|
||||||
|
|
||||||
|
rightParam, val3 := c.compileIfVarsub(rightOperand, varholder, templateName, holdreflect)
|
||||||
|
if val3.IsValid() {
|
||||||
|
val = val3
|
||||||
|
} else {
|
||||||
|
panic("val3 is invalid")
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Refactor this
|
||||||
|
out = "w.Write([]byte(common.BuildWidget(" + leftParam + "," + rightParam + ")))\n"
|
||||||
|
literal = true
|
||||||
|
break ArgLoop
|
||||||
default:
|
default:
|
||||||
c.log("Variable!")
|
c.log("Variable!")
|
||||||
if len(node.Args) > (pos + 1) {
|
if len(node.Args) > (pos + 1) {
|
||||||
|
@ -635,7 +668,7 @@ ArgLoop:
|
||||||
out += c.compileIfVarsubN(id.String(), varholder, templateName, holdreflect)
|
out += c.compileIfVarsubN(id.String(), varholder, templateName, holdreflect)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return out, val
|
return out, val, literal
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *CTemplateSet) compileReflectSwitch(varholder string, holdreflect reflect.Value, templateName string, node *parse.CommandNode) (out string, outVal reflect.Value) {
|
func (c *CTemplateSet) compileReflectSwitch(varholder string, holdreflect reflect.Value, templateName string, node *parse.CommandNode) (out string, outVal reflect.Value) {
|
||||||
|
@ -778,6 +811,12 @@ func (c *CTemplateSet) compileBoolsub(varname string, varholder string, template
|
||||||
|
|
||||||
func (c *CTemplateSet) compileVarsub(varname string, val reflect.Value, assLines string) (out string) {
|
func (c *CTemplateSet) compileVarsub(varname string, val reflect.Value, assLines string) (out string) {
|
||||||
c.log("in compileVarsub")
|
c.log("in compileVarsub")
|
||||||
|
|
||||||
|
// Is this a literal string?
|
||||||
|
if len(varname) != 0 && varname[0] == '"' {
|
||||||
|
return assLines + "w.Write([]byte(" + varname + "))\n"
|
||||||
|
}
|
||||||
|
|
||||||
for _, varItem := range c.varList {
|
for _, varItem := range c.varList {
|
||||||
if strings.HasPrefix(varname, varItem.Destination) {
|
if strings.HasPrefix(varname, varItem.Destination) {
|
||||||
varname = strings.Replace(varname, varItem.Destination, varItem.Name, 1)
|
varname = strings.Replace(varname, varItem.Destination, varItem.Name, 1)
|
||||||
|
@ -818,7 +857,7 @@ func (c *CTemplateSet) compileVarsub(varname string, val reflect.Value, assLines
|
||||||
fmt.Println("Unknown Variable Name:", varname)
|
fmt.Println("Unknown Variable Name:", varname)
|
||||||
fmt.Println("Unknown Kind:", val.Kind())
|
fmt.Println("Unknown Kind:", val.Kind())
|
||||||
fmt.Println("Unknown Type:", val.Type().Name())
|
fmt.Println("Unknown Type:", val.Type().Name())
|
||||||
panic("// I don't know what this variable's type is o.o\n")
|
panic("-- I don't know what this variable's type is o.o\n")
|
||||||
}
|
}
|
||||||
c.log("out: ", out)
|
c.log("out: ", out)
|
||||||
return assLines + out
|
return assLines + out
|
||||||
|
@ -826,7 +865,7 @@ func (c *CTemplateSet) compileVarsub(varname string, val reflect.Value, assLines
|
||||||
|
|
||||||
func (c *CTemplateSet) compileSubtemplate(pvarholder string, pholdreflect reflect.Value, node *parse.TemplateNode) (out string) {
|
func (c *CTemplateSet) compileSubtemplate(pvarholder string, pholdreflect reflect.Value, node *parse.TemplateNode) (out string) {
|
||||||
c.log("in compileSubtemplate")
|
c.log("in compileSubtemplate")
|
||||||
c.log("Template Node:", node.Name)
|
c.log("Template Node: ", node.Name)
|
||||||
|
|
||||||
fname := strings.TrimSuffix(node.Name, filepath.Ext(node.Name))
|
fname := strings.TrimSuffix(node.Name, filepath.Ext(node.Name))
|
||||||
varholder := "tmpl_" + fname + "_vars"
|
varholder := "tmpl_" + fname + "_vars"
|
||||||
|
@ -841,8 +880,8 @@ func (c *CTemplateSet) compileSubtemplate(pvarholder string, pholdreflect reflec
|
||||||
case *parse.NilNode:
|
case *parse.NilNode:
|
||||||
panic("Nil is not a command x.x")
|
panic("Nil is not a command x.x")
|
||||||
default:
|
default:
|
||||||
out = "var " + varholder + " := false\n"
|
c.log("Unknown Node: ", firstWord)
|
||||||
out += c.compileCommand(cmd)
|
panic("")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -883,16 +922,18 @@ func (c *CTemplateSet) log(args ...interface{}) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *CTemplateSet) logf(left string, args ...interface{}) {
|
||||||
|
if c.superDebug {
|
||||||
|
fmt.Printf(left, args...)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (c *CTemplateSet) error(args ...interface{}) {
|
func (c *CTemplateSet) error(args ...interface{}) {
|
||||||
if c.debug {
|
if c.debug {
|
||||||
fmt.Println(args...)
|
fmt.Println(args...)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *CTemplateSet) compileCommand(*parse.CommandNode) (out string) {
|
|
||||||
panic("Uh oh! Something went wrong!")
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: Write unit tests for this
|
// TODO: Write unit tests for this
|
||||||
func minify(data string) string {
|
func minify(data string) string {
|
||||||
data = strings.Replace(data, "\t", "", -1)
|
data = strings.Replace(data, "\t", "", -1)
|
||||||
|
|
|
@ -30,25 +30,22 @@ var ChangeDefaultThemeMutex sync.Mutex
|
||||||
|
|
||||||
// TODO: Use this when the default theme doesn't exist
|
// TODO: Use this when the default theme doesn't exist
|
||||||
var fallbackTheme = "shadow"
|
var fallbackTheme = "shadow"
|
||||||
|
|
||||||
//var overridenTemplates map[string]interface{} = make(map[string]interface{})
|
|
||||||
var overridenTemplates = make(map[string]bool)
|
var overridenTemplates = make(map[string]bool)
|
||||||
|
|
||||||
type Theme struct {
|
type Theme struct {
|
||||||
Name string
|
Name string
|
||||||
FriendlyName string
|
FriendlyName string
|
||||||
Version string
|
Version string
|
||||||
Creator string
|
Creator string
|
||||||
FullImage string
|
FullImage string
|
||||||
MobileFriendly bool
|
MobileFriendly bool
|
||||||
Disabled bool
|
Disabled bool
|
||||||
HideFromThemes bool
|
HideFromThemes bool
|
||||||
ForkOf string
|
ForkOf string
|
||||||
Tag string
|
Tag string
|
||||||
URL string
|
URL string
|
||||||
Sidebars string // Allowed Values: left, right, both, false
|
Docks []string // Allowed Values: leftSidebar, rightSidebar, footer
|
||||||
AboutSegment bool // ? - Should this be a theme var instead?
|
AboutSegment bool // ? - Should this be a theme var instead?
|
||||||
//DisableMinifier // Is this really a good idea? I don't think themes should be fighting against the minifier
|
|
||||||
Settings map[string]ThemeSetting
|
Settings map[string]ThemeSetting
|
||||||
Templates []TemplateMapping
|
Templates []TemplateMapping
|
||||||
TemplatesMap map[string]string
|
TemplatesMap map[string]string
|
||||||
|
@ -117,12 +114,12 @@ func (themes ThemeList) LoadActiveStatus() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
if defaultThemeSwitch {
|
if defaultThemeSwitch {
|
||||||
log.Print("Loading the default theme '" + theme.Name + "'")
|
log.Printf("Loading the default theme '%s'", theme.Name)
|
||||||
theme.Active = true
|
theme.Active = true
|
||||||
DefaultThemeBox.Store(theme.Name)
|
DefaultThemeBox.Store(theme.Name)
|
||||||
MapThemeTemplates(theme)
|
MapThemeTemplates(theme)
|
||||||
} else {
|
} else {
|
||||||
log.Print("Loading the theme '" + theme.Name + "'")
|
log.Printf("Loading the theme '%s'", theme.Name)
|
||||||
theme.Active = false
|
theme.Active = false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -144,7 +141,7 @@ func InitThemes() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
themeName := themeFile.Name()
|
themeName := themeFile.Name()
|
||||||
log.Print("Adding theme '" + themeName + "'")
|
log.Printf("Adding theme '%s'", themeName)
|
||||||
themeFile, err := ioutil.ReadFile("./themes/" + themeName + "/theme.json")
|
themeFile, err := ioutil.ReadFile("./themes/" + themeName + "/theme.json")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -172,9 +169,7 @@ func InitThemes() error {
|
||||||
}
|
}
|
||||||
|
|
||||||
if theme.FullImage != "" {
|
if theme.FullImage != "" {
|
||||||
if Dev.DebugMode {
|
debugLog("Adding theme image")
|
||||||
log.Print("Adding theme image")
|
|
||||||
}
|
|
||||||
err = StaticFiles.Add("./themes/"+themeName+"/"+theme.FullImage, "./themes/"+themeName)
|
err = StaticFiles.Add("./themes/"+themeName+"/"+theme.FullImage, "./themes/"+themeName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -207,9 +202,7 @@ func InitThemes() error {
|
||||||
func AddThemeStaticFiles(theme Theme) error {
|
func AddThemeStaticFiles(theme Theme) error {
|
||||||
// TODO: Use a function instead of a closure to make this more testable? What about a function call inside the closure to take the theme variable into account?
|
// TODO: Use a function instead of a closure to make this more testable? What about a function call inside the closure to take the theme variable into account?
|
||||||
return filepath.Walk("./themes/"+theme.Name+"/public", func(path string, f os.FileInfo, err error) error {
|
return filepath.Walk("./themes/"+theme.Name+"/public", func(path string, f os.FileInfo, err error) error {
|
||||||
if Dev.DebugMode {
|
debugLog("Attempting to add static file '" + path + "' for default theme '" + theme.Name + "'")
|
||||||
log.Print("Attempting to add static file '" + path + "' for default theme '" + theme.Name + "'")
|
|
||||||
}
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -224,13 +217,10 @@ func AddThemeStaticFiles(theme Theme) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
var ext = filepath.Ext(path)
|
var ext = filepath.Ext(path)
|
||||||
//log.Print("path ",path)
|
|
||||||
//log.Print("ext ",ext)
|
|
||||||
if ext == ".css" && len(data) != 0 {
|
if ext == ".css" && len(data) != 0 {
|
||||||
var b bytes.Buffer
|
var b bytes.Buffer
|
||||||
var pieces = strings.Split(path, "/")
|
var pieces = strings.Split(path, "/")
|
||||||
var filename = pieces[len(pieces)-1]
|
var filename = pieces[len(pieces)-1]
|
||||||
//log.Print("filename ", filename)
|
|
||||||
err = theme.ResourceTemplates.ExecuteTemplate(&b, filename, CSSData{ComingSoon: "We don't have any data to pass you yet!"})
|
err = theme.ResourceTemplates.ExecuteTemplate(&b, filename, CSSData{ComingSoon: "We don't have any data to pass you yet!"})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
@ -242,9 +232,7 @@ func AddThemeStaticFiles(theme Theme) error {
|
||||||
gzipData := compressBytesGzip(data)
|
gzipData := compressBytesGzip(data)
|
||||||
StaticFiles["/static/"+theme.Name+path] = SFile{data, gzipData, 0, int64(len(data)), int64(len(gzipData)), mime.TypeByExtension(ext), f, f.ModTime().UTC().Format(http.TimeFormat)}
|
StaticFiles["/static/"+theme.Name+path] = SFile{data, gzipData, 0, int64(len(data)), int64(len(gzipData)), mime.TypeByExtension(ext), f, f.ModTime().UTC().Format(http.TimeFormat)}
|
||||||
|
|
||||||
if Dev.DebugMode {
|
debugLog("Added the '/" + theme.Name + path + "' static file for theme " + theme.Name + ".")
|
||||||
log.Print("Added the '/" + theme.Name + path + "' static file for theme " + theme.Name + ".")
|
|
||||||
}
|
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -349,14 +337,12 @@ func ResetTemplateOverrides() {
|
||||||
|
|
||||||
originPointer, ok := TmplPtrMap["o_"+name]
|
originPointer, ok := TmplPtrMap["o_"+name]
|
||||||
if !ok {
|
if !ok {
|
||||||
//log.Fatal("The origin template doesn't exist!")
|
|
||||||
log.Print("The origin template doesn't exist!")
|
log.Print("The origin template doesn't exist!")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
destTmplPtr, ok := TmplPtrMap[name]
|
destTmplPtr, ok := TmplPtrMap[name]
|
||||||
if !ok {
|
if !ok {
|
||||||
//log.Fatal("The destination template doesn't exist!")
|
|
||||||
log.Print("The destination template doesn't exist!")
|
log.Print("The destination template doesn't exist!")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -519,3 +505,12 @@ func GetDefaultThemeName() string {
|
||||||
func SetDefaultThemeName(name string) {
|
func SetDefaultThemeName(name string) {
|
||||||
DefaultThemeBox.Store(name)
|
DefaultThemeBox.Store(name)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (theme Theme) HasDock(name string) bool {
|
||||||
|
for _, dock := range theme.Docks {
|
||||||
|
if dock == name {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
|
@ -5,7 +5,9 @@ import (
|
||||||
"bytes"
|
"bytes"
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"html/template"
|
||||||
"log"
|
"log"
|
||||||
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"../query_gen/lib"
|
"../query_gen/lib"
|
||||||
|
@ -15,9 +17,10 @@ var Docks WidgetDocks
|
||||||
var widgetUpdateMutex sync.RWMutex
|
var widgetUpdateMutex sync.RWMutex
|
||||||
|
|
||||||
type WidgetDocks struct {
|
type WidgetDocks struct {
|
||||||
LeftSidebar []Widget
|
LeftSidebar []*Widget
|
||||||
RightSidebar []Widget
|
RightSidebar []*Widget
|
||||||
//PanelLeft []Menus
|
//PanelLeft []Menus
|
||||||
|
Footer []*Widget
|
||||||
}
|
}
|
||||||
|
|
||||||
type Widget struct {
|
type Widget struct {
|
||||||
|
@ -25,6 +28,9 @@ type Widget struct {
|
||||||
Location string // Coming Soon: overview, topics, topic / topic_view, forums, forum, global
|
Location string // Coming Soon: overview, topics, topic / topic_view, forums, forum, global
|
||||||
Position int
|
Position int
|
||||||
Body string
|
Body string
|
||||||
|
Side string
|
||||||
|
Type string
|
||||||
|
Literal bool
|
||||||
}
|
}
|
||||||
|
|
||||||
type WidgetMenu struct {
|
type WidgetMenu struct {
|
||||||
|
@ -40,7 +46,7 @@ type WidgetMenuItem struct {
|
||||||
|
|
||||||
type NameTextPair struct {
|
type NameTextPair struct {
|
||||||
Name string
|
Name string
|
||||||
Text string
|
Text template.HTML
|
||||||
}
|
}
|
||||||
|
|
||||||
type WidgetStmts struct {
|
type WidgetStmts struct {
|
||||||
|
@ -58,6 +64,113 @@ func init() {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func preparseWidget(widget *Widget, wdata string) (err error) {
|
||||||
|
prebuildWidget := func(name string, data interface{}) (string, error) {
|
||||||
|
var b bytes.Buffer
|
||||||
|
err := Templates.ExecuteTemplate(&b, name+".html", data)
|
||||||
|
return string(b.Bytes()), err
|
||||||
|
}
|
||||||
|
|
||||||
|
sbytes := []byte(wdata)
|
||||||
|
switch widget.Type {
|
||||||
|
case "simple":
|
||||||
|
var tmp NameTextPair
|
||||||
|
err = json.Unmarshal(sbytes, &tmp)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
widget.Literal = true
|
||||||
|
widget.Body, err = prebuildWidget("widget_simple", tmp)
|
||||||
|
case "about":
|
||||||
|
var tmp NameTextPair
|
||||||
|
err = json.Unmarshal(sbytes, &tmp)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
widget.Literal = true
|
||||||
|
widget.Body, err = prebuildWidget("widget_about", tmp)
|
||||||
|
default:
|
||||||
|
widget.Literal = true
|
||||||
|
widget.Body = wdata
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Test this
|
||||||
|
// TODO: Should we toss this through a proper parser rather than crudely replacing it?
|
||||||
|
widget.Location = strings.Replace(widget.Location, " ", "", -1)
|
||||||
|
widget.Location = strings.Replace(widget.Location, "frontend", "!panel", -1)
|
||||||
|
widget.Location = strings.Replace(widget.Location, "!!", "", -1)
|
||||||
|
|
||||||
|
// Skip blank zones
|
||||||
|
var locs = strings.Split(widget.Location, "|")
|
||||||
|
if len(locs) > 0 {
|
||||||
|
widget.Location = ""
|
||||||
|
for _, loc := range locs {
|
||||||
|
if loc == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
widget.Location += loc + "|"
|
||||||
|
}
|
||||||
|
widget.Location = widget.Location[:len(widget.Location)-1]
|
||||||
|
}
|
||||||
|
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func BuildWidget(dock string, headerVars *HeaderVars) (sbody string) {
|
||||||
|
var widgets []*Widget
|
||||||
|
if !headerVars.Theme.HasDock(dock) {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
switch dock {
|
||||||
|
case "rightSidebar":
|
||||||
|
widgets = Docks.RightSidebar
|
||||||
|
case "footer":
|
||||||
|
widgets = Docks.Footer
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, widget := range widgets {
|
||||||
|
if !widget.Enabled {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if widget.Allowed(headerVars.Zone) {
|
||||||
|
item, err := widget.Build(headerVars)
|
||||||
|
if err != nil {
|
||||||
|
LogError(err)
|
||||||
|
}
|
||||||
|
sbody += item
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return sbody
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Test this
|
||||||
|
// TODO: Add support for zone:id. Perhaps, carry a ZoneID property around in headerVars? It might allow some weirdness like frontend[5] which matches any zone with an ID of 5 but it would be a tad faster than verifying each zone, although it might be problematic if users end up relying on this behaviour for areas which don't pass IDs to the widgets system but *probably* should
|
||||||
|
func (widget *Widget) Allowed(zone string) bool {
|
||||||
|
for _, loc := range strings.Split(widget.Location, "|") {
|
||||||
|
if loc == "global" || loc == zone {
|
||||||
|
return true
|
||||||
|
} else if len(loc) > 0 && loc[0] == '!' {
|
||||||
|
loc = loc[1:]
|
||||||
|
if loc != "global" && loc != zone {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Refactor
|
||||||
|
func (widget *Widget) Build(hvars interface{}) (string, error) {
|
||||||
|
if widget.Literal {
|
||||||
|
return widget.Body, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var b bytes.Buffer
|
||||||
|
var headerVars = hvars.(*HeaderVars)
|
||||||
|
err := RunThemeTemplate(headerVars.Theme.Name, widget.Body, hvars, headerVars.Writer)
|
||||||
|
return string(b.Bytes()), err
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: Make a store for this?
|
// TODO: Make a store for this?
|
||||||
func InitWidgets() error {
|
func InitWidgets() error {
|
||||||
rows, err := widgetStmts.getWidgets.Query()
|
rows, err := widgetStmts.getWidgets.Query()
|
||||||
|
@ -66,42 +179,30 @@ func InitWidgets() error {
|
||||||
}
|
}
|
||||||
defer rows.Close()
|
defer rows.Close()
|
||||||
|
|
||||||
var sbytes []byte
|
var data string
|
||||||
var side, wtype, data string
|
var leftWidgets []*Widget
|
||||||
|
var rightWidgets []*Widget
|
||||||
var leftWidgets []Widget
|
var footerWidgets []*Widget
|
||||||
var rightWidgets []Widget
|
|
||||||
|
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
var widget Widget
|
var widget = &Widget{Position: 0}
|
||||||
err = rows.Scan(&widget.Position, &side, &wtype, &widget.Enabled, &widget.Location, &data)
|
err = rows.Scan(&widget.Position, &widget.Side, &widget.Type, &widget.Enabled, &widget.Location, &data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
sbytes = []byte(data)
|
err = preparseWidget(widget, data)
|
||||||
switch wtype {
|
if err != nil {
|
||||||
case "simple":
|
return err
|
||||||
var tmp NameTextPair
|
|
||||||
err = json.Unmarshal(sbytes, &tmp)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
var b bytes.Buffer
|
|
||||||
err = Templates.ExecuteTemplate(&b, "widget_simple.html", tmp)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
widget.Body = string(b.Bytes())
|
|
||||||
default:
|
|
||||||
widget.Body = data
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if side == "left" {
|
switch widget.Side {
|
||||||
|
case "left":
|
||||||
leftWidgets = append(leftWidgets, widget)
|
leftWidgets = append(leftWidgets, widget)
|
||||||
} else if side == "right" {
|
case "right":
|
||||||
rightWidgets = append(rightWidgets, widget)
|
rightWidgets = append(rightWidgets, widget)
|
||||||
|
case "footer":
|
||||||
|
footerWidgets = append(footerWidgets, widget)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
err = rows.Err()
|
err = rows.Err()
|
||||||
|
@ -112,11 +213,13 @@ func InitWidgets() error {
|
||||||
widgetUpdateMutex.Lock()
|
widgetUpdateMutex.Lock()
|
||||||
Docks.LeftSidebar = leftWidgets
|
Docks.LeftSidebar = leftWidgets
|
||||||
Docks.RightSidebar = rightWidgets
|
Docks.RightSidebar = rightWidgets
|
||||||
|
Docks.Footer = footerWidgets
|
||||||
widgetUpdateMutex.Unlock()
|
widgetUpdateMutex.Unlock()
|
||||||
|
|
||||||
if Dev.SuperDebug {
|
if Dev.SuperDebug {
|
||||||
log.Print("Docks.LeftSidebar", Docks.LeftSidebar)
|
log.Print("Docks.LeftSidebar", Docks.LeftSidebar)
|
||||||
log.Print("Docks.RightSidebar", Docks.RightSidebar)
|
log.Print("Docks.RightSidebar", Docks.RightSidebar)
|
||||||
|
log.Print("Docks.Footer", Docks.Footer)
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
|
|
|
@ -112,6 +112,7 @@ func PrebuildTmplList(user common.User, headerVars *common.HeaderVars) common.CT
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Do this properly via the widget system
|
// TODO: Do this properly via the widget system
|
||||||
|
// TODO: REWRITE THIS
|
||||||
func CommonAreaWidgets(headerVars *common.HeaderVars) {
|
func CommonAreaWidgets(headerVars *common.HeaderVars) {
|
||||||
// TODO: Hot Groups? Featured Groups? Official Groups?
|
// TODO: Hot Groups? Featured Groups? Official Groups?
|
||||||
var b bytes.Buffer
|
var b bytes.Buffer
|
||||||
|
@ -125,9 +126,9 @@ func CommonAreaWidgets(headerVars *common.HeaderVars) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if common.Themes[headerVars.Theme.Name].Sidebars == "left" {
|
if headerVars.Theme.HasDock("leftSidebar") {
|
||||||
headerVars.Widgets.LeftSidebar = template.HTML(string(b.Bytes()))
|
headerVars.Widgets.LeftSidebar = template.HTML(string(b.Bytes()))
|
||||||
} else if common.Themes[headerVars.Theme.Name].Sidebars == "right" || common.Themes[headerVars.Theme.Name].Sidebars == "both" {
|
} else if headerVars.Theme.HasDock("rightSidebar") {
|
||||||
headerVars.Widgets.RightSidebar = template.HTML(string(b.Bytes()))
|
headerVars.Widgets.RightSidebar = template.HTML(string(b.Bytes()))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -28,6 +28,9 @@ go get -u github.com/robertkrimen/otto
|
||||||
echo "Installing the Rez Image Resizer"
|
echo "Installing the Rez Image Resizer"
|
||||||
go get -u github.com/bamiaux/rez
|
go get -u github.com/bamiaux/rez
|
||||||
|
|
||||||
|
echo "Installing fsnotify"
|
||||||
|
go get -u github.com/fsnotify/fsnotify
|
||||||
|
|
||||||
|
|
||||||
echo "Building the installer"
|
echo "Building the installer"
|
||||||
cd ./install
|
cd ./install
|
||||||
|
|
|
@ -85,6 +85,13 @@ if %errorlevel% neq 0 (
|
||||||
exit /b %errorlevel%
|
exit /b %errorlevel%
|
||||||
)
|
)
|
||||||
|
|
||||||
|
echo Installing fsnotify
|
||||||
|
go get -u github.com/fsnotify/fsnotify
|
||||||
|
if %errorlevel% neq 0 (
|
||||||
|
pause
|
||||||
|
exit /b %errorlevel%
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
echo Building the installer
|
echo Building the installer
|
||||||
go generate
|
go generate
|
||||||
|
|
14
main.go
14
main.go
|
@ -12,6 +12,8 @@ import (
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"os/signal"
|
||||||
|
"syscall"
|
||||||
"time"
|
"time"
|
||||||
//"runtime/pprof"
|
//"runtime/pprof"
|
||||||
"./common"
|
"./common"
|
||||||
|
@ -144,6 +146,7 @@ func main() {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
}
|
}
|
||||||
|
defer db.Close()
|
||||||
|
|
||||||
err = afterDBInit()
|
err = afterDBInit()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -158,7 +161,7 @@ func main() {
|
||||||
// Run this goroutine once a second
|
// Run this goroutine once a second
|
||||||
secondTicker := time.NewTicker(1 * time.Second)
|
secondTicker := time.NewTicker(1 * time.Second)
|
||||||
fifteenMinuteTicker := time.NewTicker(15 * time.Minute)
|
fifteenMinuteTicker := time.NewTicker(15 * time.Minute)
|
||||||
//hour_ticker := time.NewTicker(1 * time.Hour)
|
//hourTicker := time.NewTicker(1 * time.Hour)
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
|
@ -231,13 +234,18 @@ func main() {
|
||||||
router.HandleFunc("/profile/reply/edit/submit/", routeProfileReplyEditSubmit)
|
router.HandleFunc("/profile/reply/edit/submit/", routeProfileReplyEditSubmit)
|
||||||
router.HandleFunc("/profile/reply/delete/submit/", routeProfileReplyDeleteSubmit)
|
router.HandleFunc("/profile/reply/delete/submit/", routeProfileReplyDeleteSubmit)
|
||||||
//router.HandleFunc("/user/edit/submit/", routeLogout) // routeLogout? what on earth? o.o
|
//router.HandleFunc("/user/edit/submit/", routeLogout) // routeLogout? what on earth? o.o
|
||||||
//router.HandleFunc("/exit/", routeExit)
|
|
||||||
router.HandleFunc("/ws/", routeWebsockets)
|
router.HandleFunc("/ws/", routeWebsockets)
|
||||||
|
|
||||||
log.Print("Initialising the plugins")
|
log.Print("Initialising the plugins")
|
||||||
common.InitPlugins()
|
common.InitPlugins()
|
||||||
|
|
||||||
defer db.Close()
|
sigs := make(chan os.Signal, 1)
|
||||||
|
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
|
||||||
|
go func() {
|
||||||
|
sig := <-sigs
|
||||||
|
// TODO: Gracefully shutdown the HTTP server
|
||||||
|
log.Fatal("Received a signal to shutdown: ", sig)
|
||||||
|
}()
|
||||||
|
|
||||||
//if profiling {
|
//if profiling {
|
||||||
// pprof.StopCPUProfile()
|
// pprof.StopCPUProfile()
|
||||||
|
|
|
@ -41,8 +41,8 @@ func routeTopicCreate(w http.ResponseWriter, r *http.Request, user common.User,
|
||||||
if !user.Perms.ViewTopic || !user.Perms.CreateTopic {
|
if !user.Perms.ViewTopic || !user.Perms.CreateTopic {
|
||||||
return common.NoPermissions(w, r, user)
|
return common.NoPermissions(w, r, user)
|
||||||
}
|
}
|
||||||
|
headerVars.Zone = "create_topic"
|
||||||
common.BuildWidgets("create_topic", nil, headerVars, r)
|
headerVars.Writer = w
|
||||||
|
|
||||||
// Lock this to the forum being linked?
|
// Lock this to the forum being linked?
|
||||||
// Should we always put it in strictmode when it's linked from another forum? Well, the user might end up changing their mind on what forum they want to post in and it would be a hassle, if they had to switch pages, even if it is a single click for many (exc. mobile)
|
// Should we always put it in strictmode when it's linked from another forum? Well, the user might end up changing their mind on what forum they want to post in and it would be a hassle, if they had to switch pages, even if it is a single click for many (exc. mobile)
|
||||||
|
|
105
panel_routes.go
105
panel_routes.go
|
@ -1614,6 +1614,60 @@ func routePanelBackups(w http.ResponseWriter, r *http.Request, user common.User,
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: Log errors when something really screwy is going on?
|
||||||
|
func handleUnknownUser(user *common.User, err error) *common.User {
|
||||||
|
if err != nil {
|
||||||
|
return &common.User{Name: "Unknown", Link: common.BuildProfileURL("unknown", 0)}
|
||||||
|
}
|
||||||
|
return user
|
||||||
|
}
|
||||||
|
func handleUnknownTopic(topic *common.Topic, err error) *common.Topic {
|
||||||
|
if err != nil {
|
||||||
|
return &common.Topic{Title: "Unknown", Link: common.BuildProfileURL("unknown", 0)}
|
||||||
|
}
|
||||||
|
return topic
|
||||||
|
}
|
||||||
|
|
||||||
|
func modlogsElementType(action string, elementType string, elementID int, actor *common.User) (out string) {
|
||||||
|
switch elementType {
|
||||||
|
case "topic":
|
||||||
|
topic := handleUnknownTopic(common.Topics.Get(elementID))
|
||||||
|
switch action {
|
||||||
|
case "lock":
|
||||||
|
out = "<a href='%s'>%s</a> was locked by <a href='%s'>%s</a>"
|
||||||
|
case "unlock":
|
||||||
|
out = "<a href='%s'>%s</a> was reopened by <a href='%s'>%s</a>"
|
||||||
|
case "stick":
|
||||||
|
out = "<a href='%s'>%s</a> was pinned by <a href='%s'>%s</a>"
|
||||||
|
case "unstick":
|
||||||
|
out = "<a href='%s'>%s</a> was unpinned by <a href='%s'>%s</a>"
|
||||||
|
case "delete":
|
||||||
|
return fmt.Sprintf("Topic #%d was deleted by <a href='%s'>%s</a>", elementID, actor.Link, actor.Name)
|
||||||
|
}
|
||||||
|
out = fmt.Sprintf(out, topic.Link, topic.Title, actor.Link, actor.Name)
|
||||||
|
case "user":
|
||||||
|
targetUser := handleUnknownUser(common.Users.Get(elementID))
|
||||||
|
switch action {
|
||||||
|
case "ban":
|
||||||
|
out = "<a href='%s'>%s</a> was banned by <a href='%s'>%s</a>"
|
||||||
|
case "unban":
|
||||||
|
out = "<a href='%s'>%s</a> was unbanned by <a href='%s'>%s</a>"
|
||||||
|
case "activate":
|
||||||
|
out = "<a href='%s'>%s</a> was activated by <a href='%s'>%s</a>"
|
||||||
|
}
|
||||||
|
out = fmt.Sprintf(out, targetUser.Link, targetUser.Name, actor.Link, actor.Name)
|
||||||
|
case "reply":
|
||||||
|
if action == "delete" {
|
||||||
|
topic := handleUnknownTopic(common.BlankReply(elementID).Topic())
|
||||||
|
out = fmt.Sprintf("A reply in <a href='%s'>%s</a> was deleted by <a href='%s'>%s</a>", topic.Link, topic.Title, actor.Link, actor.Name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if out == "" {
|
||||||
|
out = fmt.Sprintf("Unknown action '%s' on elementType '%s' by <a href='%s'>%s</a>", action, elementType, actor.Link, actor.Name)
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
func routePanelLogsMod(w http.ResponseWriter, r *http.Request, user common.User) common.RouteError {
|
func routePanelLogsMod(w http.ResponseWriter, r *http.Request, user common.User) common.RouteError {
|
||||||
headerVars, stats, ferr := common.PanelUserCheck(w, r, &user)
|
headerVars, stats, ferr := common.PanelUserCheck(w, r, &user)
|
||||||
if ferr != nil {
|
if ferr != nil {
|
||||||
|
@ -1631,21 +1685,6 @@ func routePanelLogsMod(w http.ResponseWriter, r *http.Request, user common.User)
|
||||||
}
|
}
|
||||||
defer rows.Close()
|
defer rows.Close()
|
||||||
|
|
||||||
// TODO: Log errors when something really screwy is going on?
|
|
||||||
handleUnknownUser := func(user *common.User, err error) *common.User {
|
|
||||||
if err != nil {
|
|
||||||
return &common.User{Name: "Unknown", Link: common.BuildProfileURL("unknown", 0)}
|
|
||||||
}
|
|
||||||
return user
|
|
||||||
}
|
|
||||||
|
|
||||||
handleUnknownTopic := func(topic *common.Topic, err error) *common.Topic {
|
|
||||||
if err != nil {
|
|
||||||
return &common.Topic{Title: "Unknown", Link: common.BuildProfileURL("unknown", 0)}
|
|
||||||
}
|
|
||||||
return topic
|
|
||||||
}
|
|
||||||
|
|
||||||
var logs []common.LogItem
|
var logs []common.LogItem
|
||||||
var action, elementType, ipaddress, doneAt string
|
var action, elementType, ipaddress, doneAt string
|
||||||
var elementID, actorID int
|
var elementID, actorID int
|
||||||
|
@ -1656,41 +1695,7 @@ func routePanelLogsMod(w http.ResponseWriter, r *http.Request, user common.User)
|
||||||
}
|
}
|
||||||
|
|
||||||
actor := handleUnknownUser(common.Users.Get(actorID))
|
actor := handleUnknownUser(common.Users.Get(actorID))
|
||||||
|
action = modlogsElementType(action, elementType, elementID, actor)
|
||||||
switch action {
|
|
||||||
case "lock":
|
|
||||||
topic := handleUnknownTopic(common.Topics.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was locked by <a href='%s'>%s</a>", topic.Link, topic.Title, actor.Link, actor.Name)
|
|
||||||
case "unlock":
|
|
||||||
topic := handleUnknownTopic(common.Topics.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was reopened by <a href='%s'>%s</a>", topic.Link, topic.Title, actor.Link, actor.Name)
|
|
||||||
case "stick":
|
|
||||||
topic := handleUnknownTopic(common.Topics.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was pinned by <a href='%s'>%s</a>", topic.Link, topic.Title, actor.Link, actor.Name)
|
|
||||||
case "unstick":
|
|
||||||
topic := handleUnknownTopic(common.Topics.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was unpinned by <a href='%s'>%s</a>", topic.Link, topic.Title, actor.Link, actor.Name)
|
|
||||||
case "delete":
|
|
||||||
if elementType == "topic" {
|
|
||||||
action = fmt.Sprintf("Topic #%d was deleted by <a href='%s'>%s</a>", elementID, actor.Link, actor.Name)
|
|
||||||
} else {
|
|
||||||
reply := common.BlankReply()
|
|
||||||
reply.ID = elementID
|
|
||||||
topic := handleUnknownTopic(reply.Topic())
|
|
||||||
action = fmt.Sprintf("A reply in <a href='%s'>%s</a> was deleted by <a href='%s'>%s</a>", topic.Link, topic.Title, actor.Link, actor.Name)
|
|
||||||
}
|
|
||||||
case "ban":
|
|
||||||
targetUser := handleUnknownUser(common.Users.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was banned by <a href='%s'>%s</a>", targetUser.Link, targetUser.Name, actor.Link, actor.Name)
|
|
||||||
case "unban":
|
|
||||||
targetUser := handleUnknownUser(common.Users.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was unbanned by <a href='%s'>%s</a>", targetUser.Link, targetUser.Name, actor.Link, actor.Name)
|
|
||||||
case "activate":
|
|
||||||
targetUser := handleUnknownUser(common.Users.Get(elementID))
|
|
||||||
action = fmt.Sprintf("<a href='%s'>%s</a> was activated by <a href='%s'>%s</a>", targetUser.Link, targetUser.Name, actor.Link, actor.Name)
|
|
||||||
default:
|
|
||||||
action = fmt.Sprintf("Unknown action '%s' by <a href='%s'>%s</a>", action, actor.Link, actor.Name)
|
|
||||||
}
|
|
||||||
logs = append(logs, common.LogItem{Action: template.HTML(action), IPAddress: ipaddress, DoneAt: doneAt})
|
logs = append(logs, common.LogItem{Action: template.HTML(action), IPAddress: ipaddress, DoneAt: doneAt})
|
||||||
}
|
}
|
||||||
err = rows.Err()
|
err = rows.Err()
|
||||||
|
|
|
@ -115,8 +115,6 @@ func seedTables(adapter qgen.Adapter) error {
|
||||||
qgen.Install.SimpleInsert("settings", "name, content, type, constraints", "'activation_type','1','list','1-3'")
|
qgen.Install.SimpleInsert("settings", "name, content, type, constraints", "'activation_type','1','list','1-3'")
|
||||||
qgen.Install.SimpleInsert("settings", "name, content, type", "'bigpost_min_words','250','int'")
|
qgen.Install.SimpleInsert("settings", "name, content, type", "'bigpost_min_words','250','int'")
|
||||||
qgen.Install.SimpleInsert("settings", "name, content, type", "'megapost_min_words','1000','int'")
|
qgen.Install.SimpleInsert("settings", "name, content, type", "'megapost_min_words','1000','int'")
|
||||||
qgen.Install.SimpleInsert("settings", "name, content, type", "'about_segment_title','','text'")
|
|
||||||
qgen.Install.SimpleInsert("settings", "name, content, type", "'about_segment_body','','text'")
|
|
||||||
qgen.Install.SimpleInsert("themes", "uname, default", "'tempra-simple',1")
|
qgen.Install.SimpleInsert("themes", "uname, default", "'tempra-simple',1")
|
||||||
qgen.Install.SimpleInsert("emails", "email, uid, validated", "'admin@localhost',1,1") // ? - Use a different default email or let the admin input it during installation?
|
qgen.Install.SimpleInsert("emails", "email, uid, validated", "'admin@localhost',1,1") // ? - Use a different default email or let the admin input it during installation?
|
||||||
|
|
||||||
|
|
20
routes.go
20
routes.go
|
@ -101,7 +101,8 @@ func routeOverview(w http.ResponseWriter, r *http.Request, user common.User) com
|
||||||
if ferr != nil {
|
if ferr != nil {
|
||||||
return ferr
|
return ferr
|
||||||
}
|
}
|
||||||
common.BuildWidgets("overview", nil, headerVars, r)
|
headerVars.Zone = "overview"
|
||||||
|
headerVars.Writer = w
|
||||||
|
|
||||||
pi := common.Page{common.GetTitlePhrase("overview"), user, headerVars, tList, nil}
|
pi := common.Page{common.GetTitlePhrase("overview"), user, headerVars, tList, nil}
|
||||||
if common.PreRenderHooks["pre_render_overview"] != nil {
|
if common.PreRenderHooks["pre_render_overview"] != nil {
|
||||||
|
@ -127,7 +128,8 @@ func routeCustomPage(w http.ResponseWriter, r *http.Request, user common.User) c
|
||||||
if common.Templates.Lookup("page_"+name) == nil {
|
if common.Templates.Lookup("page_"+name) == nil {
|
||||||
return common.NotFound(w, r)
|
return common.NotFound(w, r)
|
||||||
}
|
}
|
||||||
common.BuildWidgets("custom_page", name, headerVars, r)
|
headerVars.Zone = "custom_page"
|
||||||
|
headerVars.Writer = w
|
||||||
|
|
||||||
pi := common.Page{common.GetTitlePhrase("page"), user, headerVars, tList, nil}
|
pi := common.Page{common.GetTitlePhrase("page"), user, headerVars, tList, nil}
|
||||||
if common.PreRenderHooks["pre_render_custom_page"] != nil {
|
if common.PreRenderHooks["pre_render_custom_page"] != nil {
|
||||||
|
@ -148,7 +150,8 @@ func routeTopics(w http.ResponseWriter, r *http.Request, user common.User) commo
|
||||||
if ferr != nil {
|
if ferr != nil {
|
||||||
return ferr
|
return ferr
|
||||||
}
|
}
|
||||||
common.BuildWidgets("topics", nil, headerVars, r)
|
headerVars.Zone = "topics"
|
||||||
|
headerVars.Writer = w
|
||||||
|
|
||||||
// TODO: Add a function for the qlist stuff
|
// TODO: Add a function for the qlist stuff
|
||||||
var qlist string
|
var qlist string
|
||||||
|
@ -333,8 +336,8 @@ func routeForum(w http.ResponseWriter, r *http.Request, user common.User, sfid s
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
return common.InternalError(err, w, r)
|
return common.InternalError(err, w, r)
|
||||||
}
|
}
|
||||||
|
headerVars.Zone = "view_forum"
|
||||||
common.BuildWidgets("view_forum", forum, headerVars, r)
|
headerVars.Writer = w
|
||||||
|
|
||||||
// Calculate the offset
|
// Calculate the offset
|
||||||
var offset int
|
var offset int
|
||||||
|
@ -420,7 +423,8 @@ func routeForums(w http.ResponseWriter, r *http.Request, user common.User) commo
|
||||||
if ferr != nil {
|
if ferr != nil {
|
||||||
return ferr
|
return ferr
|
||||||
}
|
}
|
||||||
common.BuildWidgets("forums", nil, headerVars, r)
|
headerVars.Zone = "forums"
|
||||||
|
headerVars.Writer = w
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
var forumList []common.Forum
|
var forumList []common.Forum
|
||||||
|
@ -509,8 +513,8 @@ func routeTopicID(w http.ResponseWriter, r *http.Request, user common.User) comm
|
||||||
//log.Printf("user.Perms: %+v\n", user.Perms)
|
//log.Printf("user.Perms: %+v\n", user.Perms)
|
||||||
return common.NoPermissions(w, r, user)
|
return common.NoPermissions(w, r, user)
|
||||||
}
|
}
|
||||||
|
headerVars.Zone = "view_topic"
|
||||||
common.BuildWidgets("view_topic", &topic, headerVars, r)
|
headerVars.Writer = w
|
||||||
|
|
||||||
topic.ContentHTML = common.ParseMessage(topic.Content, topic.ParentID, "forums")
|
topic.ContentHTML = common.ParseMessage(topic.Content, topic.ParentID, "forums")
|
||||||
topic.ContentLines = strings.Count(topic.Content, "\n")
|
topic.ContentLines = strings.Count(topic.Content, "\n")
|
||||||
|
|
|
@ -3,8 +3,6 @@ INSERT INTO [settings] ([name],[content],[type]) VALUES ('url_tags','1','bool');
|
||||||
INSERT INTO [settings] ([name],[content],[type],[constraints]) VALUES ('activation_type','1','list','1-3');
|
INSERT INTO [settings] ([name],[content],[type],[constraints]) VALUES ('activation_type','1','list','1-3');
|
||||||
INSERT INTO [settings] ([name],[content],[type]) VALUES ('bigpost_min_words','250','int');
|
INSERT INTO [settings] ([name],[content],[type]) VALUES ('bigpost_min_words','250','int');
|
||||||
INSERT INTO [settings] ([name],[content],[type]) VALUES ('megapost_min_words','1000','int');
|
INSERT INTO [settings] ([name],[content],[type]) VALUES ('megapost_min_words','1000','int');
|
||||||
INSERT INTO [settings] ([name],[content],[type]) VALUES ('about_segment_title','','text');
|
|
||||||
INSERT INTO [settings] ([name],[content],[type]) VALUES ('about_segment_body','','text');
|
|
||||||
INSERT INTO [themes] ([uname],[default]) VALUES ('tempra-simple',1);
|
INSERT INTO [themes] ([uname],[default]) VALUES ('tempra-simple',1);
|
||||||
INSERT INTO [emails] ([email],[uid],[validated]) VALUES ('admin@localhost',1,1);
|
INSERT INTO [emails] ([email],[uid],[validated]) VALUES ('admin@localhost',1,1);
|
||||||
INSERT INTO [users_groups] ([name],[permissions],[plugin_perms],[is_mod],[is_admin],[tag]) VALUES ('Administrator','{"BanUsers":true,"ActivateUsers":true,"EditUser":true,"EditUserEmail":true,"EditUserPassword":true,"EditUserGroup":true,"EditUserGroupSuperMod":true,"EditUserGroupAdmin":false,"EditGroup":true,"EditGroupLocalPerms":true,"EditGroupGlobalPerms":true,"EditGroupSuperMod":true,"EditGroupAdmin":false,"ManageForums":true,"EditSettings":true,"ManageThemes":true,"ManagePlugins":true,"ViewAdminLogs":true,"ViewIPs":true,"UploadFiles":true,"ViewTopic":true,"LikeItem":true,"CreateTopic":true,"EditTopic":true,"DeleteTopic":true,"CreateReply":true,"EditReply":true,"DeleteReply":true,"PinTopic":true,"CloseTopic":true}','{}',1,1,'Admin');
|
INSERT INTO [users_groups] ([name],[permissions],[plugin_perms],[is_mod],[is_admin],[tag]) VALUES ('Administrator','{"BanUsers":true,"ActivateUsers":true,"EditUser":true,"EditUserEmail":true,"EditUserPassword":true,"EditUserGroup":true,"EditUserGroupSuperMod":true,"EditUserGroupAdmin":false,"EditGroup":true,"EditGroupLocalPerms":true,"EditGroupGlobalPerms":true,"EditGroupSuperMod":true,"EditGroupAdmin":false,"ManageForums":true,"EditSettings":true,"ManageThemes":true,"ManagePlugins":true,"ViewAdminLogs":true,"ViewIPs":true,"UploadFiles":true,"ViewTopic":true,"LikeItem":true,"CreateTopic":true,"EditTopic":true,"DeleteTopic":true,"CreateReply":true,"EditReply":true,"DeleteReply":true,"PinTopic":true,"CloseTopic":true}','{}',1,1,'Admin');
|
||||||
|
|
|
@ -3,8 +3,6 @@ INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('url_tags','1','bool');
|
||||||
INSERT INTO `settings`(`name`,`content`,`type`,`constraints`) VALUES ('activation_type','1','list','1-3');
|
INSERT INTO `settings`(`name`,`content`,`type`,`constraints`) VALUES ('activation_type','1','list','1-3');
|
||||||
INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('bigpost_min_words','250','int');
|
INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('bigpost_min_words','250','int');
|
||||||
INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('megapost_min_words','1000','int');
|
INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('megapost_min_words','1000','int');
|
||||||
INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('about_segment_title','','text');
|
|
||||||
INSERT INTO `settings`(`name`,`content`,`type`) VALUES ('about_segment_body','','text');
|
|
||||||
INSERT INTO `themes`(`uname`,`default`) VALUES ('tempra-simple',1);
|
INSERT INTO `themes`(`uname`,`default`) VALUES ('tempra-simple',1);
|
||||||
INSERT INTO `emails`(`email`,`uid`,`validated`) VALUES ('admin@localhost',1,1);
|
INSERT INTO `emails`(`email`,`uid`,`validated`) VALUES ('admin@localhost',1,1);
|
||||||
INSERT INTO `users_groups`(`name`,`permissions`,`plugin_perms`,`is_mod`,`is_admin`,`tag`) VALUES ('Administrator','{"BanUsers":true,"ActivateUsers":true,"EditUser":true,"EditUserEmail":true,"EditUserPassword":true,"EditUserGroup":true,"EditUserGroupSuperMod":true,"EditUserGroupAdmin":false,"EditGroup":true,"EditGroupLocalPerms":true,"EditGroupGlobalPerms":true,"EditGroupSuperMod":true,"EditGroupAdmin":false,"ManageForums":true,"EditSettings":true,"ManageThemes":true,"ManagePlugins":true,"ViewAdminLogs":true,"ViewIPs":true,"UploadFiles":true,"ViewTopic":true,"LikeItem":true,"CreateTopic":true,"EditTopic":true,"DeleteTopic":true,"CreateReply":true,"EditReply":true,"DeleteReply":true,"PinTopic":true,"CloseTopic":true}','{}',1,1,'Admin');
|
INSERT INTO `users_groups`(`name`,`permissions`,`plugin_perms`,`is_mod`,`is_admin`,`tag`) VALUES ('Administrator','{"BanUsers":true,"ActivateUsers":true,"EditUser":true,"EditUserEmail":true,"EditUserPassword":true,"EditUserGroup":true,"EditUserGroupSuperMod":true,"EditUserGroupAdmin":false,"EditGroup":true,"EditGroupLocalPerms":true,"EditGroupGlobalPerms":true,"EditGroupSuperMod":true,"EditGroupAdmin":false,"ManageForums":true,"EditSettings":true,"ManageThemes":true,"ManagePlugins":true,"ViewAdminLogs":true,"ViewIPs":true,"UploadFiles":true,"ViewTopic":true,"LikeItem":true,"CreateTopic":true,"EditTopic":true,"DeleteTopic":true,"CreateReply":true,"EditReply":true,"DeleteReply":true,"PinTopic":true,"CloseTopic":true}','{}',1,1,'Admin');
|
||||||
|
|
|
@ -27,5 +27,3 @@
|
||||||
;
|
;
|
||||||
;
|
;
|
||||||
;
|
;
|
||||||
;
|
|
||||||
;
|
|
||||||
|
|
|
@ -203,37 +203,26 @@ w.Write(forum_57)
|
||||||
w.Write(forum_58)
|
w.Write(forum_58)
|
||||||
}
|
}
|
||||||
w.Write(forum_59)
|
w.Write(forum_59)
|
||||||
if tmpl_forum_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_forum_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_forum_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_forum_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_forum_vars.Header.Themes) != 0 {
|
if len(tmpl_forum_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_forum_vars.Header.Themes {
|
for _, item := range tmpl_forum_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_forum_vars.Header.Theme.Name == item.Name {
|
if tmpl_forum_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_forum_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_forum_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_forum_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -118,37 +118,26 @@ w.Write(forums_18)
|
||||||
w.Write(forums_19)
|
w.Write(forums_19)
|
||||||
}
|
}
|
||||||
w.Write(forums_20)
|
w.Write(forums_20)
|
||||||
if tmpl_forums_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_forums_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_forums_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_forums_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_forums_vars.Header.Themes) != 0 {
|
if len(tmpl_forums_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_forums_vars.Header.Themes {
|
for _, item := range tmpl_forums_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_forums_vars.Header.Theme.Name == item.Name {
|
if tmpl_forums_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_forums_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_forums_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_forums_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -91,37 +91,26 @@ w.Write(guilds_guild_list_6)
|
||||||
w.Write(guilds_guild_list_7)
|
w.Write(guilds_guild_list_7)
|
||||||
}
|
}
|
||||||
w.Write(guilds_guild_list_8)
|
w.Write(guilds_guild_list_8)
|
||||||
if tmpl_guilds_guild_list_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_guilds_guild_list_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_guilds_guild_list_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_guilds_guild_list_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_guilds_guild_list_vars.Header.Themes) != 0 {
|
if len(tmpl_guilds_guild_list_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_guilds_guild_list_vars.Header.Themes {
|
for _, item := range tmpl_guilds_guild_list_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_guilds_guild_list_vars.Header.Theme.Name == item.Name {
|
if tmpl_guilds_guild_list_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_guilds_guild_list_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_guilds_guild_list_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_guilds_guild_list_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -264,36 +264,31 @@ var topic_100 = []byte(`
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
`)
|
`)
|
||||||
var footer_0 = []byte(`<div class="about">
|
var footer_0 = []byte(`<div class="footer">
|
||||||
<a id="aboutTitle">`)
|
`)
|
||||||
var footer_1 = []byte(`</a>
|
var footer_1 = []byte(`
|
||||||
<span id="aboutDesc">`)
|
<div id="poweredByHolder" class="footerBit">
|
||||||
var footer_2 = []byte(`</span>
|
<div id="poweredBy">
|
||||||
</div>`)
|
<a id="poweredByName" href="https://github.com/Azareal/Gosora">Powered by Gosora</a><span id="poweredByDash"> - </span><span id="poweredByMaker">Made with love by Azareal</span>
|
||||||
var footer_3 = []byte(`
|
|
||||||
<div class="footer">
|
|
||||||
<div id="poweredBy">
|
|
||||||
<a id="poweredByName" href="https://github.com/Azareal/Gosora">Powered by Gosora</a><span id="poweredByDash"> - </span><span id="poweredByMaker">Made with love by Azareal</span>
|
|
||||||
</div>
|
|
||||||
<form action="/theme/" method="post">
|
|
||||||
<div id="themeSelector" style="float: right;">
|
|
||||||
<select id="themeSelectorSelect" name="themeSelector" aria-label="Change the site's appearance">
|
|
||||||
`)
|
|
||||||
var footer_4 = []byte(`<option val="`)
|
|
||||||
var footer_5 = []byte(`"`)
|
|
||||||
var footer_6 = []byte(` selected`)
|
|
||||||
var footer_7 = []byte(`>`)
|
|
||||||
var footer_8 = []byte(`</option>`)
|
|
||||||
var footer_9 = []byte(`
|
|
||||||
</select>
|
|
||||||
</div>
|
</div>
|
||||||
</form>
|
<form action="/theme/" method="post">
|
||||||
|
<div id="themeSelector" style="float: right;">
|
||||||
|
<select id="themeSelectorSelect" name="themeSelector" aria-label="Change the site's appearance">
|
||||||
|
`)
|
||||||
|
var footer_2 = []byte(`<option val="`)
|
||||||
|
var footer_3 = []byte(`"`)
|
||||||
|
var footer_4 = []byte(` selected`)
|
||||||
|
var footer_5 = []byte(`>`)
|
||||||
|
var footer_6 = []byte(`</option>`)
|
||||||
|
var footer_7 = []byte(`
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
`)
|
<aside class="sidebar">`)
|
||||||
var footer_10 = []byte(`<aside class="sidebar">`)
|
var footer_8 = []byte(`</aside>
|
||||||
var footer_11 = []byte(`</aside>`)
|
|
||||||
var footer_12 = []byte(`
|
|
||||||
<div style="clear: both;"></div>
|
<div style="clear: both;"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -161,37 +161,26 @@ w.Write(profile_41)
|
||||||
}
|
}
|
||||||
w.Write(profile_42)
|
w.Write(profile_42)
|
||||||
w.Write(profile_43)
|
w.Write(profile_43)
|
||||||
if tmpl_profile_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_profile_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_profile_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_profile_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_profile_vars.Header.Themes) != 0 {
|
if len(tmpl_profile_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_profile_vars.Header.Themes {
|
for _, item := range tmpl_profile_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_profile_vars.Header.Theme.Name == item.Name {
|
if tmpl_profile_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_profile_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_profile_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_profile_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,9 @@
|
||||||
// Code generated by Gosora. More below:
|
// Code generated by Gosora. More below:
|
||||||
/* This file was automatically generated by the software. Please don't edit it as your changes may be overwritten at any moment. */
|
/* This file was automatically generated by the software. Please don't edit it as your changes may be overwritten at any moment. */
|
||||||
package main
|
package main
|
||||||
import "strconv"
|
|
||||||
import "net/http"
|
import "net/http"
|
||||||
import "./common"
|
import "./common"
|
||||||
|
import "strconv"
|
||||||
|
|
||||||
// nolint
|
// nolint
|
||||||
func init() {
|
func init() {
|
||||||
|
@ -296,37 +296,26 @@ w.Write(topic_98)
|
||||||
w.Write(topic_99)
|
w.Write(topic_99)
|
||||||
}
|
}
|
||||||
w.Write(topic_100)
|
w.Write(topic_100)
|
||||||
if tmpl_topic_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_topic_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_topic_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_topic_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_topic_vars.Header.Themes) != 0 {
|
if len(tmpl_topic_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_topic_vars.Header.Themes {
|
for _, item := range tmpl_topic_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_topic_vars.Header.Theme.Name == item.Name {
|
if tmpl_topic_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_topic_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_topic_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_topic_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,9 @@
|
||||||
// Code generated by Gosora. More below:
|
// Code generated by Gosora. More below:
|
||||||
/* This file was automatically generated by the software. Please don't edit it as your changes may be overwritten at any moment. */
|
/* This file was automatically generated by the software. Please don't edit it as your changes may be overwritten at any moment. */
|
||||||
package main
|
package main
|
||||||
import "net/http"
|
|
||||||
import "./common"
|
import "./common"
|
||||||
import "strconv"
|
import "strconv"
|
||||||
|
import "net/http"
|
||||||
|
|
||||||
// nolint
|
// nolint
|
||||||
func init() {
|
func init() {
|
||||||
|
@ -302,37 +302,26 @@ w.Write(topic_alt_102)
|
||||||
w.Write(topic_alt_103)
|
w.Write(topic_alt_103)
|
||||||
}
|
}
|
||||||
w.Write(topic_alt_104)
|
w.Write(topic_alt_104)
|
||||||
if tmpl_topic_alt_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_topic_alt_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_topic_alt_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_topic_alt_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_topic_alt_vars.Header.Themes) != 0 {
|
if len(tmpl_topic_alt_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_topic_alt_vars.Header.Themes {
|
for _, item := range tmpl_topic_alt_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_topic_alt_vars.Header.Theme.Name == item.Name {
|
if tmpl_topic_alt_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_topic_alt_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_topic_alt_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_topic_alt_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,9 @@
|
||||||
// Code generated by Gosora. More below:
|
// Code generated by Gosora. More below:
|
||||||
/* This file was automatically generated by the software. Please don't edit it as your changes may be overwritten at any moment. */
|
/* This file was automatically generated by the software. Please don't edit it as your changes may be overwritten at any moment. */
|
||||||
package main
|
package main
|
||||||
|
import "strconv"
|
||||||
import "net/http"
|
import "net/http"
|
||||||
import "./common"
|
import "./common"
|
||||||
import "strconv"
|
|
||||||
|
|
||||||
// nolint
|
// nolint
|
||||||
func init() {
|
func init() {
|
||||||
|
@ -199,37 +199,26 @@ w.Write(topics_55)
|
||||||
w.Write(topics_56)
|
w.Write(topics_56)
|
||||||
}
|
}
|
||||||
w.Write(topics_57)
|
w.Write(topics_57)
|
||||||
if tmpl_topics_vars.Header.Theme.AboutSegment {
|
|
||||||
w.Write(footer_0)
|
w.Write(footer_0)
|
||||||
dispInt := tmpl_topics_vars.Header.Settings["about_segment_title"]
|
w.Write([]byte(common.BuildWidget("footer",tmpl_topics_vars.Header)))
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_1)
|
w.Write(footer_1)
|
||||||
dispInt = tmpl_topics_vars.Header.Settings["about_segment_body"]
|
|
||||||
w.Write([]byte(dispInt.(string)))
|
|
||||||
w.Write(footer_2)
|
|
||||||
}
|
|
||||||
w.Write(footer_3)
|
|
||||||
if len(tmpl_topics_vars.Header.Themes) != 0 {
|
if len(tmpl_topics_vars.Header.Themes) != 0 {
|
||||||
for _, item := range tmpl_topics_vars.Header.Themes {
|
for _, item := range tmpl_topics_vars.Header.Themes {
|
||||||
if !item.HideFromThemes {
|
if !item.HideFromThemes {
|
||||||
w.Write(footer_4)
|
w.Write(footer_2)
|
||||||
w.Write([]byte(item.Name))
|
w.Write([]byte(item.Name))
|
||||||
w.Write(footer_5)
|
w.Write(footer_3)
|
||||||
if tmpl_topics_vars.Header.Theme.Name == item.Name {
|
if tmpl_topics_vars.Header.Theme.Name == item.Name {
|
||||||
|
w.Write(footer_4)
|
||||||
|
}
|
||||||
|
w.Write(footer_5)
|
||||||
|
w.Write([]byte(item.FriendlyName))
|
||||||
w.Write(footer_6)
|
w.Write(footer_6)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
w.Write(footer_7)
|
w.Write(footer_7)
|
||||||
w.Write([]byte(item.FriendlyName))
|
w.Write([]byte(common.BuildWidget("rightSidebar",tmpl_topics_vars.Header)))
|
||||||
w.Write(footer_8)
|
w.Write(footer_8)
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
w.Write(footer_9)
|
|
||||||
if tmpl_topics_vars.Header.Widgets.RightSidebar != "" {
|
|
||||||
w.Write(footer_10)
|
|
||||||
w.Write([]byte(string(tmpl_topics_vars.Header.Widgets.RightSidebar)))
|
|
||||||
w.Write(footer_11)
|
|
||||||
}
|
|
||||||
w.Write(footer_12)
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,23 +1,22 @@
|
||||||
{{if .Header.Theme.AboutSegment}}<div class="about">
|
|
||||||
<a id="aboutTitle">{{.Header.Settings.about_segment_title}}</a>
|
|
||||||
<span id="aboutDesc">{{.Header.Settings.about_segment_body}}</span>
|
|
||||||
</div>{{end}}
|
|
||||||
<div class="footer">
|
<div class="footer">
|
||||||
<div id="poweredBy">
|
{{dock "footer" .Header }}
|
||||||
<a id="poweredByName" href="https://github.com/Azareal/Gosora">Powered by Gosora</a><span id="poweredByDash"> - </span><span id="poweredByMaker">Made with love by Azareal</span>
|
<div id="poweredByHolder" class="footerBit">
|
||||||
</div>
|
<div id="poweredBy">
|
||||||
<form action="/theme/" method="post">
|
<a id="poweredByName" href="https://github.com/Azareal/Gosora">Powered by Gosora</a><span id="poweredByDash"> - </span><span id="poweredByMaker">Made with love by Azareal</span>
|
||||||
<div id="themeSelector" style="float: right;">
|
|
||||||
<select id="themeSelectorSelect" name="themeSelector" aria-label="Change the site's appearance">
|
|
||||||
{{range .Header.Themes}}
|
|
||||||
{{if not .HideFromThemes}}<option val="{{.Name}}"{{if eq $.Header.Theme.Name .Name}} selected{{end}}>{{.FriendlyName}}</option>{{end}}
|
|
||||||
{{end}}
|
|
||||||
</select>
|
|
||||||
</div>
|
</div>
|
||||||
</form>
|
<form action="/theme/" method="post">
|
||||||
|
<div id="themeSelector" style="float: right;">
|
||||||
|
<select id="themeSelectorSelect" name="themeSelector" aria-label="Change the site's appearance">
|
||||||
|
{{range .Header.Themes}}
|
||||||
|
{{if not .HideFromThemes}}<option val="{{.Name}}"{{if eq $.Header.Theme.Name .Name}} selected{{end}}>{{.FriendlyName}}</option>{{end}}
|
||||||
|
{{end}}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{{if .Header.Widgets.RightSidebar}}<aside class="sidebar">{{.Header.Widgets.RightSidebar}}</aside>{{end}}
|
<aside class="sidebar">{{dock "rightSidebar" .Header }}</aside>
|
||||||
<div style="clear: both;"></div>
|
<div style="clear: both;"></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -31,7 +31,7 @@
|
||||||
<div class="colstack_item colstack_head">
|
<div class="colstack_item colstack_head">
|
||||||
<div class="rowitem"><h1>Create Group</h1></div>
|
<div class="rowitem"><h1>Create Group</h1></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="colstack_item">
|
<div class="colstack_item the_form">
|
||||||
<form action="/panel/groups/create/?session={{.CurrentUser.Session}}" method="post">
|
<form action="/panel/groups/create/?session={{.CurrentUser.Session}}" method="post">
|
||||||
<div class="formrow">
|
<div class="formrow">
|
||||||
<div class="formitem formlabel"><a>Name</a></div>
|
<div class="formitem formlabel"><a>Name</a></div>
|
||||||
|
|
|
@ -28,7 +28,7 @@
|
||||||
<div class="colstack_item colstack_head">
|
<div class="colstack_item colstack_head">
|
||||||
<div class="rowitem"><h1>Add Filter</h1></div>
|
<div class="rowitem"><h1>Add Filter</h1></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="colstack_item">
|
<div class="colstack_item the_form">
|
||||||
<form action="/panel/settings/word-filters/create/?session={{.CurrentUser.Session}}" method="post">
|
<form action="/panel/settings/word-filters/create/?session={{.CurrentUser.Session}}" method="post">
|
||||||
<div class="formrow">
|
<div class="formrow">
|
||||||
<div class="formitem formlabel"><a>Find</a></div>
|
<div class="formitem formlabel"><a>Find</a></div>
|
||||||
|
|
|
@ -0,0 +1,4 @@
|
||||||
|
<div class="about widget">
|
||||||
|
<a id="aboutTitle">{{.Name}}</a>
|
||||||
|
<span id="aboutDesc">{{.Text}}</span>
|
||||||
|
</div>
|
|
@ -34,22 +34,26 @@ body {
|
||||||
margin: 0px;
|
margin: 0px;
|
||||||
color: var(--primary-text-color);
|
color: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
color: var(--primary-link-color);
|
color: var(--primary-link-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
body, #back {
|
||||||
|
/*background-color: hsl(0,0%,97%);*/
|
||||||
|
background-color: hsl(0,0%,98%);
|
||||||
|
}
|
||||||
#back {
|
#back {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
padding-top: 14px;
|
padding-top: 14px;
|
||||||
display: flex;
|
display: flex;
|
||||||
/*background-color: hsl(0,0%,97%);*/
|
|
||||||
background-color: hsl(0,0%,98%);
|
|
||||||
padding-left: 0px;
|
padding-left: 0px;
|
||||||
padding-right: 0px;
|
padding-right: 0px;
|
||||||
padding-bottom: 0px;
|
padding-bottom: 0px;
|
||||||
}
|
}
|
||||||
|
.container {
|
||||||
|
background-color: var(--element-background-color);
|
||||||
|
}
|
||||||
|
|
||||||
#main {
|
#main {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
@ -61,7 +65,7 @@ a {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
.nav {
|
.nav {
|
||||||
border-bottom: 1px solid var(--header-border-color);
|
border-bottom: 2px solid var(--header-border-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
li {
|
li {
|
||||||
|
@ -163,9 +167,11 @@ ul {
|
||||||
margin-bottom: 12px;
|
margin-bottom: 12px;
|
||||||
border: 1px solid var(--header-border-color);
|
border: 1px solid var(--header-border-color);
|
||||||
border-bottom: 2px solid var(--header-border-color);
|
border-bottom: 2px solid var(--header-border-color);
|
||||||
background-color: var(--element-background-color);
|
|
||||||
margin-left: 12px;
|
margin-left: 12px;
|
||||||
}
|
}
|
||||||
|
.rowblock:not(.topic_list), .colstack_head, .topic_row .rowitem {
|
||||||
|
background-color: var(--element-background-color);
|
||||||
|
}
|
||||||
.rowblock {
|
.rowblock {
|
||||||
margin-right: 12px;
|
margin-right: 12px;
|
||||||
}
|
}
|
||||||
|
@ -203,7 +209,7 @@ ul {
|
||||||
width: 300px;
|
width: 300px;
|
||||||
}
|
}
|
||||||
.colstack:not(#profile_container) .colstack_right {
|
.colstack:not(#profile_container) .colstack_right {
|
||||||
width: calc(90% - 300px);
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.extra_little_row_avatar {
|
.extra_little_row_avatar {
|
||||||
|
@ -470,18 +476,25 @@ select, input, textarea, button {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
}
|
}
|
||||||
|
.topic_list .topic_row:last-child .rowitem {
|
||||||
|
margin-bottom: 0px;
|
||||||
|
}
|
||||||
#forum_topic_list .topic_inner_left .starter {
|
#forum_topic_list .topic_inner_left .starter {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
width: 200px;
|
width: 200px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.topic_left, .topic_right {
|
.rowlist .rowitem, .topic_left, .topic_right {
|
||||||
margin-bottom: 8px;
|
margin-bottom: 8px;
|
||||||
padding: 4px;
|
padding: 4px;
|
||||||
display: flex;
|
display: flex;
|
||||||
border: 1px solid var(--element-border-color);
|
border: 1px solid var(--element-border-color);
|
||||||
border-bottom: 2px solid var(--element-border-color);
|
border-bottom: 2px solid var(--element-border-color);
|
||||||
}
|
}
|
||||||
|
.rowlist .rowitem {
|
||||||
|
background-color: var(--element-background-color);
|
||||||
|
padding: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
.topic_list .rowtopic {
|
.topic_list .rowtopic {
|
||||||
font-size: 17px;
|
font-size: 17px;
|
||||||
|
@ -788,9 +801,10 @@ select, input, textarea, button {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin-right: 12px;
|
margin-right: 12px;
|
||||||
}
|
}
|
||||||
#profile_right_lane .colstack_item, .colstack_right .colstack_item {
|
#profile_right_lane .colstack_item, .colstack_right .colstack_item:not(.rowlist) {
|
||||||
border: 1px solid var(--element-border-color);
|
border: 1px solid var(--element-border-color);
|
||||||
border-bottom: 2px solid var(--element-border-color);
|
border-bottom: 2px solid var(--element-border-color);
|
||||||
|
background-color: var(--element-background-color);
|
||||||
}
|
}
|
||||||
#profile_right_lane .colstack_item, .colstack_right .colstack_item, .colstack_right .colstack_grid {
|
#profile_right_lane .colstack_item, .colstack_right .colstack_item, .colstack_right .colstack_grid {
|
||||||
margin-left: 16px;
|
margin-left: 16px;
|
||||||
|
@ -836,7 +850,11 @@ select, input, textarea, button {
|
||||||
border-right: none !important;
|
border-right: none !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.about, .footer {
|
/* TODO: Make widget_about's CSS less footer centric */
|
||||||
|
.footer {
|
||||||
|
margin-top: 14px;
|
||||||
|
}
|
||||||
|
.footerBit, .footer .widget {
|
||||||
border-top: 1px solid var(--element-border-color);
|
border-top: 1px solid var(--element-border-color);
|
||||||
padding: 12px;
|
padding: 12px;
|
||||||
padding-top: 10px;
|
padding-top: 10px;
|
||||||
|
@ -846,14 +864,14 @@ select, input, textarea, button {
|
||||||
background-color: var(--element-background-color);
|
background-color: var(--element-background-color);
|
||||||
display: flex;
|
display: flex;
|
||||||
}
|
}
|
||||||
|
#poweredByHolder {
|
||||||
|
border-bottom: 2px solid var(--element-border-color);
|
||||||
|
}
|
||||||
.about, #poweredBy {
|
.about, #poweredBy {
|
||||||
font-size: 17px;
|
font-size: 17px;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
.about {
|
|
||||||
margin-top: 14px;
|
|
||||||
}
|
|
||||||
#poweredBy {
|
#poweredBy {
|
||||||
margin-right: auto;
|
margin-right: auto;
|
||||||
}
|
}
|
||||||
|
@ -861,8 +879,9 @@ select, input, textarea, button {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
}
|
}
|
||||||
#aboutTitle {
|
#aboutTitle {
|
||||||
font-size: 18px;
|
font-size: 17px;
|
||||||
margin: 8px;
|
margin: 8px;
|
||||||
|
margin-bottom: 4px;
|
||||||
}
|
}
|
||||||
#poweredByName {
|
#poweredByName {
|
||||||
font-size: 17px;
|
font-size: 17px;
|
||||||
|
@ -872,12 +891,17 @@ select, input, textarea, button {
|
||||||
margin-left: 8px;
|
margin-left: 8px;
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
width: 60%;
|
width: 60%;
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
#aboutDesc p {
|
||||||
|
-webkit-margin-before: 12px;
|
||||||
|
-webkit-margin-after: 12px;
|
||||||
}
|
}
|
||||||
#aboutDesc p:last-child {
|
#aboutDesc p:last-child {
|
||||||
-webkit-margin-after: 8px;
|
-webkit-margin-after: 8px;
|
||||||
}
|
}
|
||||||
#aboutDesc p:first-child {
|
#aboutDesc p:first-child {
|
||||||
-webkit-margin-before: 3px;
|
-webkit-margin-before: 0px;
|
||||||
}
|
}
|
||||||
#poweredByDash, #poweredByMaker {
|
#poweredByDash, #poweredByMaker {
|
||||||
display: none;
|
display: none;
|
||||||
|
@ -890,12 +914,13 @@ select, input, textarea, button {
|
||||||
.colstack_grid {
|
.colstack_grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: repeat(3, 1fr);
|
grid-template-columns: repeat(3, 1fr);
|
||||||
|
grid-gap: 8px;
|
||||||
}
|
}
|
||||||
.grid_item {
|
.grid_item {
|
||||||
background: var(--element-background-color);
|
background: var(--element-background-color);
|
||||||
border: 1px solid var(--element-border-color);
|
border: 1px solid var(--element-border-color);
|
||||||
border-bottom: 2px solid var(--element-border-color);
|
border-bottom: 2px solid var(--element-border-color);
|
||||||
margin: 8px;
|
margin: 0px;
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
padding-left: 0px;
|
padding-left: 0px;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
@ -919,7 +944,6 @@ select, input, textarea, button {
|
||||||
padding-top: 16px;
|
padding-top: 16px;
|
||||||
padding-right: 19px;
|
padding-right: 19px;
|
||||||
color: hsl(0,0%,20%);
|
color: hsl(0,0%,20%);
|
||||||
border-bottom: 1px solid var(--element-border-color);
|
|
||||||
}
|
}
|
||||||
#dash-version:before {
|
#dash-version:before {
|
||||||
content: "\f126";
|
content: "\f126";
|
||||||
|
@ -936,13 +960,3 @@ select, input, textarea, button {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/*#dash-cpu:before {
|
|
||||||
content: "\f2db";
|
|
||||||
width: 30px;
|
|
||||||
margin-right: 8px;
|
|
||||||
display: inline-block;
|
|
||||||
padding-left: 20px;
|
|
||||||
background: hsl(0,0%,98%);
|
|
||||||
font: normal normal normal 14px/1 FontAwesome;
|
|
||||||
}*/
|
|
|
@ -1,6 +1,3 @@
|
||||||
.about {
|
.about {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
.footer {
|
|
||||||
margin-top: 14px;
|
|
||||||
}
|
|
|
@ -5,7 +5,7 @@
|
||||||
"Creator": "Azareal",
|
"Creator": "Azareal",
|
||||||
"URL": "github.com/Azareal/Gosora",
|
"URL": "github.com/Azareal/Gosora",
|
||||||
"Tag": "WIP",
|
"Tag": "WIP",
|
||||||
"Sidebars":"right",
|
"Docks":["rightSidebar","footer"],
|
||||||
"AboutSegment":true,
|
"AboutSegment":true,
|
||||||
"Templates": [
|
"Templates": [
|
||||||
{
|
{
|
||||||
|
|
|
@ -511,7 +511,7 @@ input, select, textarea {
|
||||||
padding-left: 30px;
|
padding-left: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
#poweredByHolder {
|
||||||
background-color: var(--main-block-color);
|
background-color: var(--main-block-color);
|
||||||
margin-top: 5px;
|
margin-top: 5px;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
|
@ -521,7 +521,7 @@ input, select, textarea {
|
||||||
clear: left;
|
clear: left;
|
||||||
height: 25px;
|
height: 25px;
|
||||||
}
|
}
|
||||||
.footer select {
|
#poweredByHolder select {
|
||||||
background-color: var(--input-background-color);
|
background-color: var(--input-background-color);
|
||||||
border: 1px solid var(--input-border-color);
|
border: 1px solid var(--input-border-color);
|
||||||
color: var(--input-text-color);
|
color: var(--input-text-color);
|
||||||
|
|
|
@ -860,7 +860,7 @@ button.username {
|
||||||
margin-bottom: 6px !important;
|
margin-bottom: 6px !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
#poweredByHolder {
|
||||||
border: 1px solid var(--main-border-color);
|
border: 1px solid var(--main-border-color);
|
||||||
margin-top: 12px;
|
margin-top: 12px;
|
||||||
clear: both;
|
clear: both;
|
||||||
|
@ -871,7 +871,7 @@ button.username {
|
||||||
background-color: white;
|
background-color: white;
|
||||||
border-bottom: 1.5px inset var(--main-border-color);
|
border-bottom: 1.5px inset var(--main-border-color);
|
||||||
}
|
}
|
||||||
.footer select {
|
#poweredByHolder select {
|
||||||
padding: 2px;
|
padding: 2px;
|
||||||
margin-top: 1px;
|
margin-top: 1px;
|
||||||
}
|
}
|
||||||
|
@ -902,13 +902,13 @@ button.username {
|
||||||
|
|
||||||
/* Firefox specific CSS */
|
/* Firefox specific CSS */
|
||||||
@supports (-moz-appearance: none) {
|
@supports (-moz-appearance: none) {
|
||||||
.footer, .rowmenu, #profile_right_lane .topic_reply_form, .content_container {
|
#poweredByHolder, .rowmenu, #profile_right_lane .topic_reply_form, .content_container {
|
||||||
border-bottom: 2px inset hsl(0,0%,40%);
|
border-bottom: 2px inset hsl(0,0%,40%);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* Edge... We can't get the exact shade here, because of how they implemented it x.x */
|
/* Edge... We can't get the exact shade here, because of how they implemented it x.x */
|
||||||
@supports (-ms-ime-align:auto) {
|
@supports (-ms-ime-align:auto) {
|
||||||
.footer, .rowmenu, #profile_right_lane .topic_reply_form, .content_container {
|
#poweredByHolder, .rowmenu, #profile_right_lane .topic_reply_form, .content_container {
|
||||||
border-bottom: 1.5px inset hsl(0,0%,100%);
|
border-bottom: 1.5px inset hsl(0,0%,100%);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
"FullImage": "tempra-conflux.png",
|
"FullImage": "tempra-conflux.png",
|
||||||
"MobileFriendly": true,
|
"MobileFriendly": true,
|
||||||
"URL": "github.com/Azareal/Gosora",
|
"URL": "github.com/Azareal/Gosora",
|
||||||
"Sidebars":"right",
|
"Docks":["rightSidebar"],
|
||||||
"Templates": [
|
"Templates": [
|
||||||
{
|
{
|
||||||
"Name": "topic",
|
"Name": "topic",
|
||||||
|
|
|
@ -8,5 +8,5 @@
|
||||||
"MobileFriendly": true,
|
"MobileFriendly": true,
|
||||||
"HideFromThemes": true,
|
"HideFromThemes": true,
|
||||||
"URL": "github.com/Azareal/Gosora",
|
"URL": "github.com/Azareal/Gosora",
|
||||||
"Sidebars":"right"
|
"Docks":["rightSidebar"]
|
||||||
}
|
}
|
||||||
|
|
|
@ -718,7 +718,7 @@ button.username {
|
||||||
top: -2px;
|
top: -2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
#poweredByHolder {
|
||||||
border: 1px solid hsl(0, 0%, 80%);
|
border: 1px solid hsl(0, 0%, 80%);
|
||||||
margin-top: 12px;
|
margin-top: 12px;
|
||||||
clear: both;
|
clear: both;
|
||||||
|
@ -727,7 +727,7 @@ button.username {
|
||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
padding-right: 10px;
|
padding-right: 10px;
|
||||||
}
|
}
|
||||||
.footer select {
|
#poweredByHolder select {
|
||||||
padding: 2px;
|
padding: 2px;
|
||||||
margin-top: 1px;
|
margin-top: 1px;
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,7 +6,7 @@
|
||||||
"FullImage": "tempra-simple.png",
|
"FullImage": "tempra-simple.png",
|
||||||
"MobileFriendly": true,
|
"MobileFriendly": true,
|
||||||
"URL": "github.com/Azareal/Gosora",
|
"URL": "github.com/Azareal/Gosora",
|
||||||
"Sidebars":"right",
|
"Docks":["rightSidebar"],
|
||||||
"Resources": [
|
"Resources": [
|
||||||
{
|
{
|
||||||
"Name":"tempra-simple/misc.js",
|
"Name":"tempra-simple/misc.js",
|
||||||
|
|
|
@ -27,3 +27,6 @@ go get -u github.com/go-ego/riot
|
||||||
|
|
||||||
echo "Updating the Rez Image Resizer"
|
echo "Updating the Rez Image Resizer"
|
||||||
go get -u github.com/bamiaux/rez
|
go get -u github.com/bamiaux/rez
|
||||||
|
|
||||||
|
echo "Updating fsnotify"
|
||||||
|
go get -u github.com/fsnotify/fsnotify
|
|
@ -82,5 +82,12 @@ if %errorlevel% neq 0 (
|
||||||
exit /b %errorlevel%
|
exit /b %errorlevel%
|
||||||
)
|
)
|
||||||
|
|
||||||
|
echo Updating fsnotify
|
||||||
|
go get -u github.com/fsnotify/fsnotify
|
||||||
|
if %errorlevel% neq 0 (
|
||||||
|
pause
|
||||||
|
exit /b %errorlevel%
|
||||||
|
)
|
||||||
|
|
||||||
echo The dependencies were successfully updated
|
echo The dependencies were successfully updated
|
||||||
pause
|
pause
|
||||||
|
|
Loading…
Reference in New Issue