This file is indexed.

/usr/share/denemo/actions/denemo-modules/selection.scm is in denemo-data 1.1.0-1ubuntu1.

This file is owned by root:root, with mode 0o644.

The actual contents of the file can be viewed below.

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
#!
(define-module (actions denemo-modules selection)
	#:export (
		NextSelectedObjectAllStaffs
		SingleAndSelectionSwitcher
		MapToSelection
		ApplyToTaggedSelection
		SingleAndTaggedSelectionSwitcher
		NextChordInSelection
		DenemoPaste
		SchemeCopy
		ProcessSchemeCopyBufferMusObj
		SchemePaste
	)
	#:use-module (ice-9 optargs)) !#

(define (MoveToSelectionBeginningInThisStaff)
	(define staffPosition (d-GetStaff))
	(define rememberPosition (GetPosition))
	(if (d-GoToSelectionStart)
		(begin 
			(d-GoToPosition #f staffPosition #f 1)
			(if (d-IsInSelection)
				#t
				(begin (apply d-GoToPosition rememberPosition) #f))) ; something went wrong. 
		#f)) ; no selection at all. 

;Next object in selection for all staffs
(define (NextSelectedObjectAllStaffs)
	(define lastposition (GetPosition))
	(if (and (d-MarkStatus) (d-IsInSelection))
		(if (d-NextSelectedObject)
		  #t ; found one. End
		   (if (d-MoveToStaffDown) ; no selected item left in the current staff. check one down.
		  		(if (MoveToSelectionBeginningInThisStaff) 
				  	#t ; found a selection in the lower staff
				  	(begin (apply d-GoToPosition lastposition ) #f)) ; reset cursor to the last known selection position and end.
				  #f)) ; no staff below
	  #f)); no selection or cursor not in selection	  

		
;SingleAndSelectionSwitcher by Nils Gey Jan/2010
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;Automatically applies a script to a whole selection. You can give different commands or command blocks with (begin) for single items or whole selections. You can enter a complete scheme script with (),  arguments and everything you would want to run standalone. Don't forget to escape chars like  \" . You can even use a complete (begin ) block.
;But attention! SingleAndSelectionSwitcher will still try to apply the given script to each of the single items alone. If you need a script which differs completly in beaviour for single/selection you have to write your own. You have to take out the (let loop () section for this and write your own selection part there.
;The applied script itself has to take care if the command can be applied to each potential item. If you want only notes/chords/rests you have to make sure the script does not abort on other objects. Its the same as giving proper return values for a single item, just return #f if a command is not possible for an item. While a single item just returns an error if you don't do it correctly, but does no harm otherwise, a script applied to a selection will stop on that item and leaves you on half on the way.
;Return values are the return values the script itself gives.
;The third, optional, parameter can prevent an object from be processed. By default this parameter is #t so the command will be will be applied to any object in the selection and let the command itself decide what to do (or just do nothing). By giving the third optional argument you can specify additional conditions, for example with GetType. In general: Insert test conditions here, if #t the current object will be processed, otherwise it will be skipped.
;Example: (SingleAndSelectionSwitcher d-AddDot d-ToggleStaccato) ; this is nonsense, but valid. It will add a dot for single notes or toggle staccato for the whole selection.

(define* (SingleAndSelectionSwitcher commandsingle #:optional (commandselection commandsingle) (onlyFor True)) ; Amazingly commandsingle is already defined on spot so that it can be used again in the same line to define commandselection 
	(if (string? commandsingle) ; support for old scripts. They passed the complete string of scheme as parameter
		(set! commandsingle (eval-string (string-append "(lambda () " commandsingle " )"))))
	(if (string? commandselection)
		(set! commandselection (eval-string (string-append "(lambda () " commandselection " )"))))		
	(if (and DenemoPref_applytoselection (d-MarkStatus))
		(ForEachToSelection commandselection onlyFor)
		(commandsingle)))

; MapToSelection is like schemes (map) mixed with ApplyToSelection. Use a proc on all selection items and gather all proc return values in a list. You can give an optional test, only items which return #t are processed.
(define* (MapToSelection proc #:optional (onlyFor True))
	(define return (list #f))	; prepare return list
	(define (gather)
		(if (onlyFor) ; test the current item
			(append! return (list (proc))) ; execute the proc and append its return value as listmember to the returnlist
			#f))
	(if (and DenemoPref_applytoselection (d-MarkStatus)) ; only if preferences allow it and if there is a selection at all
		(begin 
			(d-PushPosition)
			(d-GoToSelectionStart)
			(gather) ; start one without selection testing. We already know we have a selection and RepeatProcWhileTest tests first which results in ignoring the first selected item.
			(RepeatProcWhileTest gather NextSelectedObjectAllStaffs) ; Use the proc/gather function on all items in the selection
			(d-PopPosition)
			(list-tail return 1))			
		#f))

;ForEachToSelection applies the command to each item in the selection. The return value is unspecified. Faster than MapToSelection.
(define* (ForEachToSelection proc #:optional (onlyFor True))
	(define (apply)
		(if (onlyFor) ; test the current item
			(proc)			
			#f))
	(if (and DenemoPref_applytoselection (d-MarkStatus)) ; only if preferences allow it and if there is a selection at all
		(begin 
			(d-PushPosition)
			(d-GoToSelectionStart)
			(apply) ; start one without selection testing. We already know we have a selection and RepeatProcWhileTest tests first which results in ignoring the first selected item.
			(RepeatProcWhileTest apply NextSelectedObjectAllStaffs) ; Use the proc/gather function on all items in the selection
			(d-PopPosition)
			(if #f #f) ; return unspecified.
			)			
		#f))

;Three functions to tag any Denemo-object. Invisible to the user or lilypond.
(define (Tag) (d-DirectivePut-object-minpixels "select" 0))
(define (Untag) (d-DirectiveDelete-object "select"))
(define (Tag?) (d-DirectiveGetForTag-object "select" ))

;Search objects which were tagged by (Tag)
(define (NextTaggedObjectAllStaffs)
	(define position (GetPosition))
	(if (FindNextObjectAllStaffs Tag?) 	
		#t
		(begin (apply d-GoToPosition position) #f)))
					
;An alternative implementation of ApplyToSelection which works with (Tag) instead of the normal selection. This allows destructive changes which would normally destroy the Denemo-selection
;;Instead of a range, like the built-in selection, every item is tagged on its own. This is slower but allows items to be changed or deleted, which is not allowed otherwise
(define (ApplyToTaggedSelection proc)
	(if (ForEachToSelection Tag) ; ForEachToSelection tests: only for selections and if preferences allow it
		(let ()
			(d-GoToSelectionStart)
			(d-UnsetMark)
			(Untag) (proc)
			(RepeatProcWhileTest
				(lambda () (Untag) (proc)) ; The action happens here. Untag makes sure that we never encounter an endless loop because the of functions that move the cursor on their own and return to the tagged item so the movement instruction see below cannot advance.
				(lambda () ; movement/test for RepeatProc which returns #t or #f
				(if (Tag?) ; if the current object is already tagged stay. This is guaranteed to only happen once because next time it will be untagged by the line above.
					#t
					(NextTaggedObjectAllStaffs)))))
		#f)) ; no selection or not allowed by preferences

;A SingleAndSelectionSwitcherVariant that works with TaggedSelection which is more robust and works for more commands, but is slower.
;; For documentation see (SingleAndSelectionSwitcher) and (ApplyToTaggedSelection)
;; Works only with real functions, no deprecated support for string-commands like the original SingleAndSelectionSwitcher
(define* (SingleAndTaggedSelectionSwitcher commandsingle #:optional (commandselection commandsingle) (onlyFor True))
	(if (and DenemoPref_applytoselection (d-MarkStatus)) ; decide if single or selection.
		(ApplyToTaggedSelection (lambda () (if (onlyFor) (commandselection))))
		(commandsingle)))
		
(define NextChordInSelection (lambda () (if (d-NextSelectedObject) 
					    (if (Music?)
			                	 #t
			                	 (NextChordInSelection))
					    #f)))
(define FirstChordInSelection (lambda () (if (d-GoToMark)
						  (if (Music?)
			                	 #t)
						  #f)))

; Paste by Nils Gey, 2011
;; Multistaff-Pasting always adds the complete part AFTER the current measure or fills any complete set of empty measures
;; Singlestaff-Pasting happens at the cursor position and will just paste whats in the clipboard
(define* (DenemoPaste #:optional (autocreatebarlines #f))
  (define (Paste)
	(define paste:multistaff? (d-GetClipObjType 1 0))
	(define paste:howmanystaffs
		(let loop ((n 0))
		(if (d-GetClipObjType (1+ n) 0)
		(loop (1+ n))
		(1+ n))))
	(define position:startmeasure (d-GetMeasure))
	(define position:startstaff (d-GetStaff))
	(define staff 0)
	(define count -1)		
	(define staffcountlist (list 0)) ; used for multistaff			
	(define (1+count!)
		(set! count (1+ count)))
	(define (1+staff!)
		(set! staff (1+ staff)))
	(define (MeasuresToPasteToEmpty?)
		(define position:measure (d-GetMeasure))	
		(not (any not (map (lambda (x) (ProbePosition None? #f (+ position:startstaff x) (1+ position:measure) 1)) staffcountlist))))
	(define (SplitMeasure!)
		(if paste:multistaff?
			(let ()
				(define position:measure (d-GetMeasure))			
				(define position:current (GetPosition))
				(for-each (lambda (x) 
					(if (d-GoToPosition #f (+ position:startstaff x) (1+ position:measure) 1)
						(d-SplitMeasure) 
						(begin 
							(d-GoToPosition #f (+ position:startstaff x) 1 1) 
							(d-MoveToEnd) (d-SplitMeasure)))) ; for staff ends
					 staffcountlist)
				(apply d-GoToPosition position:current)
				(d-MoveToMeasureRight)) ; all needed empty measures got created			
			(d-SplitMeasure))) ; singlestaff is simple split.	
	(define (Put!) 
		(if (and autocreatebarlines (not paste:multistaff?) (not (UnderfullMeasure?)) (Appending?)) ; in a single-staff with AutoCreateBarlines #t Put! will create Barlines if the current measure is full, not MeasureBreak!
			(if (d-MoveToMeasureRight)
						#t
						(SplitMeasure!)))
		(d-PutClipObj staff count)) ; nothing special here. Just paste.				
	(define (MeasureBreak!)
		(if (and autocreatebarlines (not paste:multistaff?))
			#t ; no barline should be created by paste. Let Denemo decide if a measure is full or not.
			(if  (or (> staff 0) (MeasuresToPasteToEmpty?)) ; only the first staff needs to check if the next measure is empty or not. In Multistaff the first paste-round created all necessary empty measures for all other staffs so its just straight-forward pasting of objects.
				(d-MoveToMeasureRight)
				(SplitMeasure!))))
	(define (paste! staff count)		
		(case (d-GetClipObjType staff count)
			;In order of occurence, to boost performance.
			((0) (Put!))	;note, rest, gracenote, chord
			((8) (MeasureBreak!) ) ; Measurebreak
			((15) (d-PutClipObj staff count))	;lilypond-directive
			((1) (d-PutClipObj staff count))	;tuplet open
			((2) (d-PutClipObj staff count))	;tuplet close
			((5) (d-PutClipObj staff count))	;keysignature
			((4) (d-PutClipObj staff count))	;timesignature
			((3) (d-PutClipObj staff count))	;clef						
			((7) (d-PutClipObj staff count))	;stem-directive
			((9) #f) ; staffbreak 			
			((#f) #f) ; No object left. Means "no clipboard", too.
			(else (begin (display "Error! Object to paste unknown or unsupported\n") #f))))
			
	;body
	(d-UnsetMark)
	(set! staffcountlist (iota paste:howmanystaffs))	
	(if paste:multistaff? ; check if the staff-length of all participating staffs is equal. If not append measures.
		(let ()
		(define position:current (1+ position:startmeasure))
		(d-PushPosition)
		(for-each (lambda (x) 
		 	(if (d-GoToPosition #f (+ position:startstaff x) position:current 1)
		 		#t
		 		(begin ; fill in measures up to nr. position:startmeasure
		 			(d-GoToPosition #f (+ position:startstaff x) 1 1)
		 			(d-MoveToEnd)
		 			(Repeat d-AppendMeasure (- position:current (d-GetMeasure)))))) staffcountlist) 
		(d-PopPosition)))	
	(if paste:multistaff?  ; check if the current measure in all needed staffs is empty. If not create an empty measure to start.
		(if (any not (map (lambda (x) (ProbePosition None? #f (+ position:startstaff x) position:startmeasure 1)) staffcountlist))
			(MeasureBreak!)
			(set! position:startmeasure (1- position:startmeasure)))) 
	;Do the first staff. It will stop on staffbreak or end of the clipboard.	
	(RepeatUntilFail (lambda () (1+count!) (paste! staff count)))
	(if paste:multistaff? 
		(let ()
			(define position:return (GetPosition))			
			(Repeat  ; repeat single-staff pasting for each staff > 0.	
				(lambda ()					
					(1+staff!)
					(set! count -1)
					(if (d-GoToPosition #f (+ staff position:startstaff) (1+ position:startmeasure) 1) ; if a staff down, go there. else abort.
						(RepeatUntilFail (lambda () (1+count!) (paste! staff count)))							
						"No staff left to paste to. But the beginning of the clipboard was pasted, which is probably what you wanted."))
				paste:howmanystaffs)
			(apply d-GoToPosition position:return))))	
  (if (d-GetClipObjType 0 0) (Paste)))
  
; A copy variant in Scheme
;; Save the selection in a scheme variable
;; Music? are musobj (CreateMusObj)
;;TODO: SchemeCopy and Paste are very limited and need improvement.
(define (SchemeCopy)
  ;If on an end-tuplet marker it gives you the startvalue
  (define (GetTupletFromEndTuplet)
		(let ()
		(define return #f)
			(if (equal? (d-GetType) "TUPCLOSE")
			(begin
				(d-PushPosition)
				(let loop ()
				(if (d-MoveCursorLeft)
					(if (equal? (d-GetType) "TUPOPEN")
						(set! return (d-GetTuplet))
						(loop))
					#f)) ; staff beginning		
				(d-PopPosition)))
				return))  
  ;Mainfunction to gather data.
  (define (gather)
  	(ActionChooser 
		(lambda () (CreateMusObj)) ;chords, notes, rests
		(lambda () (cons 'TUPCLOSE (GetTupletFromEndTuplet))) ; tuplet close
		(lambda () (cons 'TUPOPEN (d-GetTuplet))) ; tuplet open
		(lambda () (disp "lily")) ; lilypond directive
		(lambda () (disp "clef")) ; clefs
		(lambda () (disp "time")) ; timesignatures
		(lambda () (disp "key")) ; keysignatures
		(lambda () (disp "stem")))) ; stem directives /voice presets		
  (if (d-MarkStatus)
	(MapToSelection gather)		
	#f))
	
(define (ProcessSchemeCopyBufferMusObj musobjproc copybuffer)
	;modify the current musobj and then return the complete, altered, object for the map-list.
	(map (lambda (current) 
		(if (musobj? current) 
			(begin (musobjproc current) current) ; if museobj use the proc
			current)) ; if not just return the original object
		 copybuffer))
	
;Paste a list created by (SchemeCopy)
(define (SchemePaste listy)
  (define (insert x)
  	(cond
  	((musobj? x)  (ANS::InsertNotes (musobj.pitch x) (musobj.baseduration x) (musobj.dots x)))
  	((equal? (car x) 'TUPCLOSE) (d-EndTuplet))
  	((equal? (car x) 'TUPOPEN) (begin (d-StartTriplet) (d-SetTuplet (cdr x))))
  	
  	))
  	(for-each (lambda (x) (insert x)) listy))

;Apply the passed script to each movement of a score
(define (ForAllMovements script)
  (d-PushPosition)
  (d-GoToPosition 1 1 1 1)
  (let loop ()
    (begin
      (eval-string script)
      (if (d-NextMovement)
	  (loop))))
  (d-PopPosition))