Archived
1
0
This repository has been archived on 2025-03-03. You can view files and clone it, but cannot push or open issues or pull requests.
ponder/settings/score.go

168 lines
4.6 KiB
Go

// Copyright © 2016 Jip J. Dekker <jip@dekker.li>
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package settings
import (
"bufio"
"encoding/json"
"io/ioutil"
"os"
"path/filepath"
"strings"
"time"
log "github.com/Sirupsen/logrus"
"github.com/jjdekker/ponder/helpers"
)
const lyToken = "% PONDER"
// Score represents the settings for a specific score file
type Score struct {
Name string // The name of the score in the songbook
Categories []string `json:",omitempty"` // Categories to which the scores belong
Path string `json:",omitempty"` // The path to the scores (uncompiled) file
LastModified time.Time `json:"-"` // Time when the score source was last modified (will be set internally)
OutputPath string `json:",omitempty"` // The path on which the compiled version of the score will be placed
}
// FromJSON reads the settings of a score from a JSON file
func FromJSON(path string) (*Score, error) {
data, err := ioutil.ReadFile(path)
if err != nil {
return nil, err
}
var s Score
err = json.Unmarshal(data, &s)
if err != nil {
return nil, err
}
s.LastModified = helpers.LastModified(s.Path)
return &s, nil
}
// FromLy reads the score metadata from a lilypond file.
// The metadata should be on one line preceded by the lyToken.
// The data itself should be in json format.
func FromLy(path string) (*Score, error) {
file, err := os.Open(path)
if err != nil {
return nil, err
}
scan := bufio.NewScanner(file)
var line string
for scan.Scan() {
if strings.Contains(scan.Text(), lyToken) {
line = strings.TrimLeftFunc(scan.Text(), func(r rune) bool {
return r != '{'
})
break
}
}
file.Close()
var s Score
if line != "" {
err = json.Unmarshal([]byte(line), &s)
if err != nil {
return nil, err
}
}
if s.Name == "" {
s.Name = filepath.Base(path)[:strings.LastIndex(filepath.Base(path), ".")]
}
s.Path = path
return &s, nil
}
// CreateScore creates a json file for a score given its path
func CreateScore(path, workDir string) {
if filepath.Ext(path) != ".pdf" && filepath.Ext(path) != ".ly" {
log.WithFields(log.Fields{"path": path}).
Fatal("Unsupported file type")
}
jsonPath := path[:strings.LastIndex(path, ".")]
s := Score{Name: filepath.Base(jsonPath)}
if filepath.Dir(path) != workDir {
s.Categories = []string{filepath.Base(filepath.Dir(path))}
}
var (
data []byte
err error
)
if filepath.Ext(path) == ".ly" {
data, err = json.Marshal(s)
helpers.Check(err, "Unable to generate valid json")
data = append(append([]byte(lyToken), data...), byte('\n'))
var ly []byte
ly, err = ioutil.ReadFile(path)
helpers.Check(err, "Unable to read lilypond file")
err = ioutil.WriteFile(path, append(data, ly...), 0644)
helpers.Check(err, "Unable to write data to lilypond file")
} else {
s.Path = path
data, err = json.MarshalIndent(s, "", " ")
helpers.Check(err, "Unable to generate valid json")
err = ioutil.WriteFile(jsonPath+".json", data, 0644)
helpers.Check(err, "Unable to save json to file")
}
}
// GenerateOutputPath fills path that the compiled score will take
func (s *Score) GenerateOutputPath(opts *Settings) {
if s.OutputPath != "" {
return
}
file := filepath.Base(s.Path)
dot := strings.LastIndex(file, ".")
if dot == -1 {
log.WithFields(log.Fields{"path": s.Path}).Error("Unable to compute output path")
return
}
file = file[:dot+1] + "pdf"
s.OutputPath = filepath.Join(opts.OutputDir, file)
}
// Scores aliases a slice of scores
type Scores []Score
// ScoresByName implements sort.Interface by providing Less and using the Len and
// Swap methods of the embedded Scores value.
type ScoresByName struct{ Scores }
// Len returns the number of scores and
// implements part of the sort.Interface
func (s Scores) Len() int { return len(s) }
// Swap interchanges the position of two scores and
// implements part of the sort.Interface
func (s Scores) Swap(i, j int) { s[i], s[j] = s[j], s[i] }
// Less reports whether the name of the score on position i is
// smaller than the name of the score on position j
// and implements part of the sort.Interface
func (s ScoresByName) Less(i, j int) bool {
return s.Scores[i].Name < s.Scores[j].Name
}