Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

STCC-227 Convert Pause X beats [TN] #238

Open
wants to merge 8 commits into
base: develop
Choose a base branch
from
57 changes: 57 additions & 0 deletions src/scratchtocatrobat/converter/converter.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
from org.catrobat.catroid import ProjectManager
import org.catrobat.catroid.common as catcommon
import org.catrobat.catroid.content as catbase
from org.catrobat.catroid.content.bricks.brickspinner import PickableDrum
from org.catrobat.catroid.ui.fragment import SpriteFactory
import org.catrobat.catroid.content.bricks as catbricks
import org.catrobat.catroid.formulaeditor as catformula
Expand Down Expand Up @@ -438,6 +439,13 @@ class _ScratchToCatrobat(object):
"changeVolumeBy:": catbricks.ChangeVolumeByNBrick,
"setVolumeTo:": catbricks.SetVolumeToBrick,

# music
"drum:duration:elapsed:from:": catbricks.PlayDrumForBeatsBrick,
"noteOn:duration:elapsed:from:": catbricks.PlayNoteForBeatsBrick,
"changeTempoBy:": catbricks.ChangeTempoByNBrick,
"setTempoTo:": catbricks.SetTempoBrick,
"rest:elapsed:from:": catbricks.PauseForBeatsBrick,

# bubble bricks
"say:duration:elapsed:from:": catbricks.SayForBubbleBrick,
"say:": catbricks.SayBubbleBrick,
Expand Down Expand Up @@ -2858,6 +2866,7 @@ def _convert_delete_line_of_list_block(self):
deleteItemBrick.setFormulaWithBrickField(catbricks.Brick.BrickField.LIST_DELETE_ITEM, index_formula)

return deleteItemBrick

if position == "all":
loop_condition_formula = catrobat.create_formula_with_value(self._converted_helper_brick_or_formula_element([list_name], "lineCountOfList:"))
catr_loop_start_brick = catbricks.RepeatBrick(loop_condition_formula)
Expand Down Expand Up @@ -3436,3 +3445,51 @@ def _convert_note_block(self):
if isinstance(arg, (str, unicode)) or isinstance(arg, catformula.Formula):
return self.CatrobatClass(arg)
log.warn("Invalid argument for NoteBrick: " + arg)

@_register_handler(_block_name_to_handler_map, "drum:duration:elapsed:from:")
def _convert_play_drum_for_beats_block(self):
[drum, duration] = self.arguments

try:
drum_int = int(drum)
except ValueError:
log.debug("There is no integer in DRUM")
return

drum_selection = PickableDrum.values()[drum_int - 1]
assert drum_selection is not None

play_drum_brick = self.CatrobatClass(catformula.Formula(duration))
play_drum_brick.drumSelection = drum_selection

return play_drum_brick

@_register_handler(_block_name_to_handler_map, "noteOn:duration:elapsed:from:")
def _convert_play_note_for_beats_block(self):
[note, duration] = self.arguments

play_note_brick = self.CatrobatClass(catformula.Formula(note), catformula.Formula(duration))
return play_note_brick

@_register_handler(_block_name_to_handler_map, "changeTempoBy:")
def _convert_change_tempo_by_block(self):
[tempo] = self.arguments

change_tempo_brick = self.CatrobatClass(catformula.Formula(tempo))
return change_tempo_brick

@_register_handler(_block_name_to_handler_map, "setTempoTo:")
def _convert_set_tempo_to_block(self):
[tempo] = self.arguments

set_tempo_brick = self.CatrobatClass(catformula.Formula(tempo))
return set_tempo_brick

@_register_handler(_block_name_to_handler_map, "rest:elapsed:from:")
def _convert_rest_for_beats_block(self):
[beats] = self.arguments

rest_for_beats = self.CatrobatClass(catformula.Formula(beats))
return rest_for_beats


63 changes: 62 additions & 1 deletion src/scratchtocatrobat/converter/test_converter.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
import org.catrobat.catroid.content.bricks.Brick as catbasebrick
import org.catrobat.catroid.formulaeditor as catformula
import org.catrobat.catroid.formulaeditor.FormulaElement.ElementType as catElementType
from scratchtocatrobat.scratch.scratch3visitor.scratch2_json_format import Scratch3_2Opcodes as opcodes
import xml.etree.cElementTree as ET

from scratchtocatrobat.converter import catrobat, converter, mediaconverter
Expand Down Expand Up @@ -1417,6 +1418,67 @@ def test_can_convert_insert_at_random_position_in_list_block(self):
assert formula_tree_value.leftChild == None
assert formula_tree_value.rightChild == None

# drum:duration:elapsed:from:
def test_can_convert_play_drum_for_beats_block(self):
drum = 2
beats = 1.0
scratch_block = ["drum:duration:elapsed:from:", drum, beats]
[catr_brick] = self.block_converter._catrobat_bricks_from(scratch_block, DUMMY_CATR_SPRITE)
assert isinstance(catr_brick, catbricks.PlayDrumForBeatsBrick)

formula_tree_list_delete_item = catr_brick.getFormulaWithBrickField(catbasebrick.BrickField.PLAY_DRUM).formulaTree # @UndefinedVariable
assert formula_tree_list_delete_item.type == catformula.FormulaElement.ElementType.NUMBER
assert formula_tree_list_delete_item.value == str(beats)
assert catr_brick.drumSelection.value == 35

# noteOn:duration:elapsed:from:
def test_can_convert_play_note_for_beats_block(self):
note = 60.0
beats = 1.0
scratch_block = ["noteOn:duration:elapsed:from:", note, beats]
[catr_brick] = self.block_converter._catrobat_bricks_from(scratch_block, DUMMY_CATR_SPRITE)
assert isinstance(catr_brick, catbricks.PlayNoteForBeatsBrick)

formula_tree_list_delete_item = catr_brick.getFormulaWithBrickField(catbasebrick.BrickField.NOTE_TO_PLAY).formulaTree # @UndefinedVariable
assert formula_tree_list_delete_item.type == catformula.FormulaElement.ElementType.NUMBER
assert formula_tree_list_delete_item.value == str(note)

formula_tree_list_delete_item = catr_brick.getFormulaWithBrickField(catbasebrick.BrickField.BEATS_TO_PLAY_NOTE).formulaTree # @UndefinedVariable
assert formula_tree_list_delete_item.type == catformula.FormulaElement.ElementType.NUMBER
assert formula_tree_list_delete_item.value == str(beats)

# changeTempoBy:
def test_can_convert_change_tempo_by_block(self):
tempo = 20.0
scratch_block = ["changeTempoBy:", tempo]
[catr_brick] = self.block_converter._catrobat_bricks_from(scratch_block, DUMMY_CATR_SPRITE)
assert isinstance(catr_brick, catbricks.ChangeTempoByNBrick)

formula_tree_list_delete_item = catr_brick.getFormulaWithBrickField(catbasebrick.BrickField.TEMPO_CHANGE).formulaTree # @UndefinedVariable
assert formula_tree_list_delete_item.type == catformula.FormulaElement.ElementType.NUMBER
assert formula_tree_list_delete_item.value == str(tempo)

# setTempoTo:
def test_can_convert_set_tempo_to_block(self):
tempo = 20.0
scratch_block = ["setTempoTo:", tempo]
[catr_brick] = self.block_converter._catrobat_bricks_from(scratch_block, DUMMY_CATR_SPRITE)
assert isinstance(catr_brick, catbricks.SetTempoBrick)

formula_tree_list_delete_item = catr_brick.getFormulaWithBrickField(catbasebrick.BrickField.TEMPO).formulaTree # @UndefinedVariable
assert formula_tree_list_delete_item.type == catformula.FormulaElement.ElementType.NUMBER
assert formula_tree_list_delete_item.value == str(tempo)

def test_can_convert_rest_for_beats_block(self):
beats = 0.25
scratch_block = ["rest:elapsed:from:", beats]
[catr_brick] = self.block_converter._catrobat_bricks_from(scratch_block, DUMMY_CATR_SPRITE)
assert isinstance(catr_brick, catbricks.PauseForBeatsBrick)

formula_tree_list_delete_item = catr_brick.getFormulaWithBrickField(catbasebrick.BrickField.BEATS_TO_PAUSE).formulaTree # @UndefinedVariable
assert formula_tree_list_delete_item.type == catformula.FormulaElement.ElementType.NUMBER
assert formula_tree_list_delete_item.value == str(beats)

# deleteLine:ofList:
def test_can_convert_delete_line_from_list_by_index_block(self):
index = 2
Expand Down Expand Up @@ -1471,7 +1533,6 @@ def test_can_convert_delete_all_lines_from_list_block(self):
assert formula_tree_list_delete_item.leftChild == None
assert formula_tree_list_delete_item.rightChild == None


# setLine:ofList:to:
def test_can_convert_set_line_via_str_index_of_list_to_block(self):
index, value = "2", "1"
Expand Down
12 changes: 11 additions & 1 deletion src/scratchtocatrobat/scratch/scratch3visitor/blockmapping.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,6 @@
import looks, motion, event, sensing, sound, operator, control, data, pen
import looks, motion, event, sensing, sound, operator, control, data, pen, music
from scratchtocatrobat.scratch.scratch3visitor.scratch2_json_format import Scratch3_2Opcodes as opcodes

visitormap = {
### event blocks ####
"event_whenflagclicked" : event.visitWhenflagclicked, #tested
Expand Down Expand Up @@ -167,5 +169,13 @@
"pen_setPenShadeToNumber" : pen.visitSetPenShadeToNumber, #tested
"pen_changePenShadeBy" : pen.visitChangePenShadeByNumber, #tested
"pen_setPenHueToNumber" : pen.visitSetPenHueToNumber, #tested

opcodes.MUSIC_PLAY_DRUM_FOR_BEATS: music.visitPlayDrumForBeats,
opcodes.MUSIC_MENU_DRUM: music.visitDrumMenu,
opcodes.MUSIC_PLAY_NOTE_FOR_BEATS: music.visitPlayNoteForBeats,
opcodes.NOTE: music.visitNoteMenu,
opcodes.MUSIC_CHANGE_TEMPO: music.visitChangeTempoBy,
opcodes.MUSIC_SET_TEMPO: music.visitSetTempoTo,
opcodes.MUSIC_REST_FOR_BEATS: music.visitRestForBeats,
}

41 changes: 41 additions & 0 deletions src/scratchtocatrobat/scratch/scratch3visitor/music.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
from visitorUtil import visitGeneric
from scratchtocatrobat.tools import logger

log = logger.log

def visitPlayDrumForBeats(blockcontext):
drum = visitGeneric(blockcontext, 'DRUM')
beats = visitGeneric(blockcontext, "BEATS")
return ['drum:duration:elapsed:from:', drum, beats]

def visitDrumMenu(blockcontext):
block = blockcontext.block
return block.fields["DRUM"][0]

def visitPlayNoteForBeats(blockcontext):
note = visitGeneric(blockcontext, 'NOTE')
beats = visitGeneric(blockcontext, "BEATS")
return ['noteOn:duration:elapsed:from:', note, beats]

def visitNoteMenu(blockcontext):
block = blockcontext.block
return block.fields["NOTE"][0]

def visitChangeTempoBy(blockcontext):
tempo = visitGeneric(blockcontext, 'TEMPO')
return ['changeTempoBy:', tempo]

def visitSetTempoTo(blockcontext):
tempo = visitGeneric(blockcontext, 'TEMPO')
return ['setTempoTo:', tempo]

def visitRestForBeats(blockcontext):
beats = visitGeneric(blockcontext, "BEATS")
return ['rest:elapsed:from:', beats]







Original file line number Diff line number Diff line change
Expand Up @@ -169,6 +169,15 @@ class Scratch3_2Opcodes(object):
OPERATOR_OR = "operator_or"
OPERATOR_LENGTH = "operator_length"

# music #
MUSIC_PLAY_DRUM_FOR_BEATS = "music_playDrumForBeats"
MUSIC_MENU_DRUM = "music_menu_DRUM"
MUSIC_PLAY_NOTE_FOR_BEATS = "music_playNoteForBeats"
NOTE = "note"
MUSIC_CHANGE_TEMPO = "music_changeTempo"
MUSIC_SET_TEMPO = "music_setTempo"
MUSIC_REST_FOR_BEATS = "music_restForBeats"

# not supported block #
NOT_SUPPORTED = "not_supported_block"

Expand Down Expand Up @@ -324,6 +333,14 @@ class Scratch3_2Opcodes(object):
OPERATOR_OR: "|",
OPERATOR_LENGTH: "stringLength:",

# music #
MUSIC_PLAY_DRUM_FOR_BEATS: "drum:duration:elapsed:from:",
MUSIC_PLAY_NOTE_FOR_BEATS: "noteOn:duration:elapsed:from:",
MUSIC_CHANGE_TEMPO: "changeTempoBy:",
MUSIC_SET_TEMPO: "setTempoTo:",
MUSIC_REST_FOR_BEATS: "rest:elapsed:from:",


### not suported block ###
NOT_SUPPORTED: "note:",
}
Expand Down
2 changes: 0 additions & 2 deletions src/scratchtocatrobat/scratch/scratch3visitor/sound.py
Original file line number Diff line number Diff line change
Expand Up @@ -50,5 +50,3 @@ def visitSounds_menu(blockcontext):
block = blockcontext.block
return block.fields["SOUND_MENU"][0]


#TODO: replace music extension bricks with default ones?