siyuan/kernel/model/themes_watcher.go
2026-03-04 22:33:35 +08:00

161 lines
3.8 KiB
Go

// SiYuan - Refactor your thinking
// Copyright (c) 2020-present, b3log.org
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <https://www.gnu.org/licenses/>.
//go:build !darwin
package model
import (
"os"
"path/filepath"
"time"
"github.com/88250/gulu"
"github.com/fsnotify/fsnotify"
"github.com/siyuan-note/logging"
"github.com/siyuan-note/siyuan/kernel/util"
)
var themesWatcher *fsnotify.Watcher
func WatchThemes() {
if !isFileWatcherAvailable() {
return
}
go watchThemes()
}
func watchThemes() {
CloseWatchThemes()
themesDir := util.ThemesPath
var err error
themesWatcher, err = fsnotify.NewWatcher()
if err != nil {
logging.LogErrorf("add themes watcher for folder [%s] failed: %s", themesDir, err)
return
}
if !gulu.File.IsDir(themesDir) {
os.MkdirAll(themesDir, 0755)
}
if err = themesWatcher.Add(themesDir); err != nil {
logging.LogErrorf("add themes root watcher for folder [%s] failed: %s", themesDir, err)
CloseWatchThemes()
return
}
// 为每个子目录添加监听,以便收到 theme.css 的变更
addThemesSubdirs(themesWatcher, themesDir)
go func() {
defer logging.Recover()
var (
timer *time.Timer
lastEvent fsnotify.Event
)
timer = time.NewTimer(100 * time.Millisecond)
<-timer.C // timer should be expired at first
for {
select {
case event, ok := <-themesWatcher.Events:
if !ok {
return
}
// 新目录创建时加入监听
if event.Op&fsnotify.Create == fsnotify.Create {
if isThemesDirectSubdir(event.Name) {
if addErr := themesWatcher.Add(event.Name); addErr != nil {
logging.LogWarnf("add themes watcher for new folder [%s] failed: %s", event.Name, addErr)
}
}
}
lastEvent = event
timer.Reset(time.Millisecond * 100)
case err, ok := <-themesWatcher.Errors:
if !ok {
return
}
logging.LogErrorf("watch themes failed: %s", err)
case <-timer.C:
handleThemesEvent(lastEvent)
}
}
}()
}
// addThemesSubdirs 为 themes 下每个子目录添加监听
func addThemesSubdirs(w *fsnotify.Watcher, themesDir string) {
entries, err := os.ReadDir(themesDir)
if err != nil {
logging.LogErrorf("read themes folder failed: %s", err)
return
}
for _, e := range entries {
if !util.IsDirRegularOrSymlink(e) {
continue
}
subdir := filepath.Join(themesDir, e.Name())
if addErr := w.Add(subdir); addErr != nil {
logging.LogWarnf("add themes watcher for folder [%s] failed: %s", subdir, addErr)
}
}
}
// isThemesDirectSubdir 判断 path 是否为 themes 下的直接子目录
func isThemesDirectSubdir(path string) bool {
if !gulu.File.IsDir(path) {
return false
}
rel, err := filepath.Rel(util.ThemesPath, path)
if err != nil {
return false
}
if filepath.Base(path) != rel {
return false
}
entries, err := os.ReadDir(util.ThemesPath)
if err != nil {
return false
}
name := filepath.Base(path)
for _, e := range entries {
if e.Name() == name {
return util.IsDirRegularOrSymlink(e)
}
}
return false
}
func handleThemesEvent(event fsnotify.Event) {
if event.Op&fsnotify.Write != fsnotify.Write {
return
}
broadcastRefreshThemeIfCurrent(event.Name)
}
func CloseWatchThemes() {
if nil != themesWatcher {
themesWatcher.Close()
themesWatcher = nil
}
}