1 /* 2 * Copyright (c) 2009-2010 jMonkeyEngine 3 * All rights reserved. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions are 7 * met: 8 * 9 * * Redistributions of source code must retain the above copyright 10 * notice, this list of conditions and the following disclaimer. 11 * 12 * * Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 16 * * Neither the name of 'jMonkeyEngine' nor the names of its contributors 17 * may be used to endorse or promote products derived from this software 18 * without specific prior written permission. 19 * 20 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 21 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 22 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 23 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR 24 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, 25 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 26 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 27 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF 28 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING 29 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS 30 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 31 */ 32 33 package com.jme3.audio; 34 35 import com.jme3.asset.AssetManager; 36 import com.jme3.asset.AssetNotFoundException; 37 import com.jme3.export.InputCapsule; 38 import com.jme3.export.JmeExporter; 39 import com.jme3.export.JmeImporter; 40 import com.jme3.export.OutputCapsule; 41 import com.jme3.math.Vector3f; 42 import com.jme3.scene.Node; 43 import com.jme3.util.PlaceholderAssets; 44 import java.io.IOException; 45 import java.util.logging.Level; 46 import java.util.logging.Logger; 47 48 /** 49 * An <code>AudioNode</code> is used in jME3 for playing audio files. 50 * <br/> 51 * First, an {@link AudioNode} is loaded from file, and then assigned 52 * to an audio node for playback. Once the audio node is attached to the 53 * scene, its location will influence the position it is playing from relative 54 * to the {@link Listener}. 55 * <br/> 56 * An audio node can also play in "headspace", meaning its location 57 * or velocity does not influence how it is played. 58 * The "positional" property of an AudioNode can be set via 59 * {@link AudioNode#setPositional(boolean) }. 60 * 61 * @author normenhansen 62 * @author Kirill Vainer 63 */ 64 public class AudioNode extends Node { 65 66 protected boolean loop = false; 67 protected float volume = 1; 68 protected float pitch = 1; 69 protected float timeOffset = 0; 70 protected Filter dryFilter; 71 protected AudioKey audioKey; 72 protected transient AudioData data = null; 73 protected transient volatile Status status = Status.Stopped; 74 protected transient volatile int channel = -1; 75 protected Vector3f velocity = new Vector3f(); 76 protected boolean reverbEnabled = true; 77 protected float maxDistance = 200; // 200 meters 78 protected float refDistance = 10; // 10 meters 79 protected Filter reverbFilter; 80 private boolean directional = false; 81 protected Vector3f direction = new Vector3f(0, 0, 1); 82 protected float innerAngle = 360; 83 protected float outerAngle = 360; 84 protected boolean positional = true; 85 86 /** 87 * <code>Status</code> indicates the current status of the audio node. 88 */ 89 public enum Status { 90 /** 91 * The audio node is currently playing. This will be set if 92 * {@link AudioNode#play() } is called. 93 */ 94 Playing, 95 96 /** 97 * The audio node is currently paused. 98 */ 99 Paused, 100 101 /** 102 * The audio node is currently stopped. 103 * This will be set if {@link AudioNode#stop() } is called 104 * or the audio has reached the end of the file. 105 */ 106 Stopped, 107 } 108 109 /** 110 * Creates a new <code>AudioNode</code> without any audio data set. 111 */ AudioNode()112 public AudioNode() { 113 } 114 115 /** 116 * Creates a new <code>AudioNode</code> without any audio data set. 117 * 118 * @param audioRenderer The audio renderer to use for playing. Cannot be null. 119 * 120 * @deprecated AudioRenderer parameter is ignored. 121 */ AudioNode(AudioRenderer audioRenderer)122 public AudioNode(AudioRenderer audioRenderer) { 123 } 124 125 /** 126 * Creates a new <code>AudioNode</code> with the given data and key. 127 * 128 * @param audioRenderer The audio renderer to use for playing. Cannot be null. 129 * @param audioData The audio data contains the audio track to play. 130 * @param audioKey The audio key that was used to load the AudioData 131 * 132 * @deprecated AudioRenderer parameter is ignored. 133 */ AudioNode(AudioRenderer audioRenderer, AudioData audioData, AudioKey audioKey)134 public AudioNode(AudioRenderer audioRenderer, AudioData audioData, AudioKey audioKey) { 135 setAudioData(audioData, audioKey); 136 } 137 138 /** 139 * Creates a new <code>AudioNode</code> with the given data and key. 140 * 141 * @param audioData The audio data contains the audio track to play. 142 * @param audioKey The audio key that was used to load the AudioData 143 */ AudioNode(AudioData audioData, AudioKey audioKey)144 public AudioNode(AudioData audioData, AudioKey audioKey) { 145 setAudioData(audioData, audioKey); 146 } 147 148 /** 149 * Creates a new <code>AudioNode</code> with the given audio file. 150 * 151 * @param audioRenderer The audio renderer to use for playing. Cannot be null. 152 * @param assetManager The asset manager to use to load the audio file 153 * @param name The filename of the audio file 154 * @param stream If true, the audio will be streamed gradually from disk, 155 * otherwise, it will be buffered. 156 * @param streamCache If stream is also true, then this specifies if 157 * the stream cache is used. When enabled, the audio stream will 158 * be read entirely but not decoded, allowing features such as 159 * seeking, looping and determining duration. 160 * 161 * @deprecated AudioRenderer parameter is ignored. 162 */ AudioNode(AudioRenderer audioRenderer, AssetManager assetManager, String name, boolean stream, boolean streamCache)163 public AudioNode(AudioRenderer audioRenderer, AssetManager assetManager, String name, boolean stream, boolean streamCache) { 164 this.audioKey = new AudioKey(name, stream, streamCache); 165 this.data = (AudioData) assetManager.loadAsset(audioKey); 166 } 167 168 /** 169 * Creates a new <code>AudioNode</code> with the given audio file. 170 * 171 * @param assetManager The asset manager to use to load the audio file 172 * @param name The filename of the audio file 173 * @param stream If true, the audio will be streamed gradually from disk, 174 * otherwise, it will be buffered. 175 * @param streamCache If stream is also true, then this specifies if 176 * the stream cache is used. When enabled, the audio stream will 177 * be read entirely but not decoded, allowing features such as 178 * seeking, looping and determining duration. 179 */ AudioNode(AssetManager assetManager, String name, boolean stream, boolean streamCache)180 public AudioNode(AssetManager assetManager, String name, boolean stream, boolean streamCache) { 181 this.audioKey = new AudioKey(name, stream, streamCache); 182 this.data = (AudioData) assetManager.loadAsset(audioKey); 183 } 184 185 /** 186 * Creates a new <code>AudioNode</code> with the given audio file. 187 * 188 * @param audioRenderer The audio renderer to use for playing. Cannot be null. 189 * @param assetManager The asset manager to use to load the audio file 190 * @param name The filename of the audio file 191 * @param stream If true, the audio will be streamed gradually from disk, 192 * otherwise, it will be buffered. 193 * 194 * @deprecated AudioRenderer parameter is ignored. 195 */ AudioNode(AudioRenderer audioRenderer, AssetManager assetManager, String name, boolean stream)196 public AudioNode(AudioRenderer audioRenderer, AssetManager assetManager, String name, boolean stream) { 197 this(audioRenderer, assetManager, name, stream, false); 198 } 199 200 /** 201 * Creates a new <code>AudioNode</code> with the given audio file. 202 * 203 * @param assetManager The asset manager to use to load the audio file 204 * @param name The filename of the audio file 205 * @param stream If true, the audio will be streamed gradually from disk, 206 * otherwise, it will be buffered. 207 */ AudioNode(AssetManager assetManager, String name, boolean stream)208 public AudioNode(AssetManager assetManager, String name, boolean stream) { 209 this(assetManager, name, stream, false); 210 } 211 212 /** 213 * Creates a new <code>AudioNode</code> with the given audio file. 214 * 215 * @param audioRenderer The audio renderer to use for playing. Cannot be null. 216 * @param assetManager The asset manager to use to load the audio file 217 * @param name The filename of the audio file 218 * 219 * @deprecated AudioRenderer parameter is ignored. 220 */ AudioNode(AudioRenderer audioRenderer, AssetManager assetManager, String name)221 public AudioNode(AudioRenderer audioRenderer, AssetManager assetManager, String name) { 222 this(assetManager, name, false); 223 } 224 225 /** 226 * Creates a new <code>AudioNode</code> with the given audio file. 227 * 228 * @param assetManager The asset manager to use to load the audio file 229 * @param name The filename of the audio file 230 */ AudioNode(AssetManager assetManager, String name)231 public AudioNode(AssetManager assetManager, String name) { 232 this(assetManager, name, false); 233 } 234 getRenderer()235 protected AudioRenderer getRenderer() { 236 AudioRenderer result = AudioContext.getAudioRenderer(); 237 if( result == null ) 238 throw new IllegalStateException( "No audio renderer available, make sure call is being performed on render thread." ); 239 return result; 240 } 241 242 /** 243 * Start playing the audio. 244 */ play()245 public void play(){ 246 getRenderer().playSource(this); 247 } 248 249 /** 250 * Start playing an instance of this audio. This method can be used 251 * to play the same <code>AudioNode</code> multiple times. Note 252 * that changes to the parameters of this AudioNode will not effect the 253 * instances already playing. 254 */ playInstance()255 public void playInstance(){ 256 getRenderer().playSourceInstance(this); 257 } 258 259 /** 260 * Stop playing the audio that was started with {@link AudioNode#play() }. 261 */ stop()262 public void stop(){ 263 getRenderer().stopSource(this); 264 } 265 266 /** 267 * Pause the audio that was started with {@link AudioNode#play() }. 268 */ pause()269 public void pause(){ 270 getRenderer().pauseSource(this); 271 } 272 273 /** 274 * Do not use. 275 */ setChannel(int channel)276 public final void setChannel(int channel) { 277 if (status != Status.Stopped) { 278 throw new IllegalStateException("Can only set source id when stopped"); 279 } 280 281 this.channel = channel; 282 } 283 284 /** 285 * Do not use. 286 */ getChannel()287 public int getChannel() { 288 return channel; 289 } 290 291 /** 292 * @return The {#link Filter dry filter} that is set. 293 * @see AudioNode#setDryFilter(com.jme3.audio.Filter) 294 */ getDryFilter()295 public Filter getDryFilter() { 296 return dryFilter; 297 } 298 299 /** 300 * Set the dry filter to use for this audio node. 301 * 302 * When {@link AudioNode#setReverbEnabled(boolean) reverb} is used, 303 * the dry filter will only influence the "dry" portion of the audio, 304 * e.g. not the reverberated parts of the AudioNode playing. 305 * 306 * See the relevent documentation for the {@link Filter} to determine 307 * the effect. 308 * 309 * @param dryFilter The filter to set, or null to disable dry filter. 310 */ setDryFilter(Filter dryFilter)311 public void setDryFilter(Filter dryFilter) { 312 this.dryFilter = dryFilter; 313 if (channel >= 0) 314 getRenderer().updateSourceParam(this, AudioParam.DryFilter); 315 } 316 317 /** 318 * Set the audio data to use for the audio. Note that this method 319 * can only be called once, if for example the audio node was initialized 320 * without an {@link AudioData}. 321 * 322 * @param audioData The audio data contains the audio track to play. 323 * @param audioKey The audio key that was used to load the AudioData 324 */ setAudioData(AudioData audioData, AudioKey audioKey)325 public void setAudioData(AudioData audioData, AudioKey audioKey) { 326 if (data != null) { 327 throw new IllegalStateException("Cannot change data once its set"); 328 } 329 330 data = audioData; 331 this.audioKey = audioKey; 332 } 333 334 /** 335 * @return The {@link AudioData} set previously with 336 * {@link AudioNode#setAudioData(com.jme3.audio.AudioData, com.jme3.audio.AudioKey) } 337 * or any of the constructors that initialize the audio data. 338 */ getAudioData()339 public AudioData getAudioData() { 340 return data; 341 } 342 343 /** 344 * @return The {@link Status} of the audio node. 345 * The status will be changed when either the {@link AudioNode#play() } 346 * or {@link AudioNode#stop() } methods are called. 347 */ getStatus()348 public Status getStatus() { 349 return status; 350 } 351 352 /** 353 * Do not use. 354 */ setStatus(Status status)355 public final void setStatus(Status status) { 356 this.status = status; 357 } 358 359 /** 360 * @return True if the audio will keep looping after it is done playing, 361 * otherwise, false. 362 * @see AudioNode#setLooping(boolean) 363 */ isLooping()364 public boolean isLooping() { 365 return loop; 366 } 367 368 /** 369 * Set the looping mode for the audio node. The default is false. 370 * 371 * @param loop True if the audio should keep looping after it is done playing. 372 */ setLooping(boolean loop)373 public void setLooping(boolean loop) { 374 this.loop = loop; 375 if (channel >= 0) 376 getRenderer().updateSourceParam(this, AudioParam.Looping); 377 } 378 379 /** 380 * @return The pitch of the audio, also the speed of playback. 381 * 382 * @see AudioNode#setPitch(float) 383 */ getPitch()384 public float getPitch() { 385 return pitch; 386 } 387 388 /** 389 * Set the pitch of the audio, also the speed of playback. 390 * The value must be between 0.5 and 2.0. 391 * 392 * @param pitch The pitch to set. 393 * @throws IllegalArgumentException If pitch is not between 0.5 and 2.0. 394 */ setPitch(float pitch)395 public void setPitch(float pitch) { 396 if (pitch < 0.5f || pitch > 2.0f) { 397 throw new IllegalArgumentException("Pitch must be between 0.5 and 2.0"); 398 } 399 400 this.pitch = pitch; 401 if (channel >= 0) 402 getRenderer().updateSourceParam(this, AudioParam.Pitch); 403 } 404 405 /** 406 * @return The volume of this audio node. 407 * 408 * @see AudioNode#setVolume(float) 409 */ getVolume()410 public float getVolume() { 411 return volume; 412 } 413 414 /** 415 * Set the volume of this audio node. 416 * 417 * The volume is specified as gain. 1.0 is the default. 418 * 419 * @param volume The volume to set. 420 * @throws IllegalArgumentException If volume is negative 421 */ setVolume(float volume)422 public void setVolume(float volume) { 423 if (volume < 0f) { 424 throw new IllegalArgumentException("Volume cannot be negative"); 425 } 426 427 this.volume = volume; 428 if (channel >= 0) 429 getRenderer().updateSourceParam(this, AudioParam.Volume); 430 } 431 432 /** 433 * @return The time offset in seconds when the sound will start playing. 434 */ getTimeOffset()435 public float getTimeOffset() { 436 return timeOffset; 437 } 438 439 /** 440 * Set the time offset in seconds when the sound will start playing. 441 * 442 * @param timeOffset The time offset 443 * @throws IllegalArgumentException If timeOffset is negative 444 */ setTimeOffset(float timeOffset)445 public void setTimeOffset(float timeOffset) { 446 if (timeOffset < 0f) { 447 throw new IllegalArgumentException("Time offset cannot be negative"); 448 } 449 450 this.timeOffset = timeOffset; 451 if (data instanceof AudioStream) { 452 System.out.println("request setTime"); 453 ((AudioStream) data).setTime(timeOffset); 454 }else if(status == Status.Playing){ 455 stop(); 456 play(); 457 } 458 } 459 460 /** 461 * @return The velocity of the audio node. 462 * 463 * @see AudioNode#setVelocity(com.jme3.math.Vector3f) 464 */ getVelocity()465 public Vector3f getVelocity() { 466 return velocity; 467 } 468 469 /** 470 * Set the velocity of the audio node. The velocity is expected 471 * to be in meters. Does nothing if the audio node is not positional. 472 * 473 * @param velocity The velocity to set. 474 * @see AudioNode#setPositional(boolean) 475 */ setVelocity(Vector3f velocity)476 public void setVelocity(Vector3f velocity) { 477 this.velocity.set(velocity); 478 if (channel >= 0) 479 getRenderer().updateSourceParam(this, AudioParam.Velocity); 480 } 481 482 /** 483 * @return True if reverb is enabled, otherwise false. 484 * 485 * @see AudioNode#setReverbEnabled(boolean) 486 */ isReverbEnabled()487 public boolean isReverbEnabled() { 488 return reverbEnabled; 489 } 490 491 /** 492 * Set to true to enable reverberation effects for this audio node. 493 * Does nothing if the audio node is not positional. 494 * <br/> 495 * When enabled, the audio environment set with 496 * {@link AudioRenderer#setEnvironment(com.jme3.audio.Environment) } 497 * will apply a reverb effect to the audio playing from this audio node. 498 * 499 * @param reverbEnabled True to enable reverb. 500 */ setReverbEnabled(boolean reverbEnabled)501 public void setReverbEnabled(boolean reverbEnabled) { 502 this.reverbEnabled = reverbEnabled; 503 if (channel >= 0) 504 getRenderer().updateSourceParam(this, AudioParam.ReverbEnabled); 505 } 506 507 /** 508 * @return Filter for the reverberations of this audio node. 509 * 510 * @see AudioNode#setReverbFilter(com.jme3.audio.Filter) 511 */ getReverbFilter()512 public Filter getReverbFilter() { 513 return reverbFilter; 514 } 515 516 /** 517 * Set the reverb filter for this audio node. 518 * <br/> 519 * The reverb filter will influence the reverberations 520 * of the audio node playing. This only has an effect if 521 * reverb is enabled. 522 * 523 * @param reverbFilter The reverb filter to set. 524 * @see AudioNode#setDryFilter(com.jme3.audio.Filter) 525 */ setReverbFilter(Filter reverbFilter)526 public void setReverbFilter(Filter reverbFilter) { 527 this.reverbFilter = reverbFilter; 528 if (channel >= 0) 529 getRenderer().updateSourceParam(this, AudioParam.ReverbFilter); 530 } 531 532 /** 533 * @return Max distance for this audio node. 534 * 535 * @see AudioNode#setMaxDistance(float) 536 */ getMaxDistance()537 public float getMaxDistance() { 538 return maxDistance; 539 } 540 541 /** 542 * Set the maximum distance for the attenuation of the audio node. 543 * Does nothing if the audio node is not positional. 544 * <br/> 545 * The maximum distance is the distance beyond which the audio 546 * node will no longer be attenuated. Normal attenuation is logarithmic 547 * from refDistance (it reduces by half when the distance doubles). 548 * Max distance sets where this fall-off stops and the sound will never 549 * get any quieter than at that distance. If you want a sound to fall-off 550 * very quickly then set ref distance very short and leave this distance 551 * very long. 552 * 553 * @param maxDistance The maximum playing distance. 554 * @throws IllegalArgumentException If maxDistance is negative 555 */ setMaxDistance(float maxDistance)556 public void setMaxDistance(float maxDistance) { 557 if (maxDistance < 0) { 558 throw new IllegalArgumentException("Max distance cannot be negative"); 559 } 560 561 this.maxDistance = maxDistance; 562 if (channel >= 0) 563 getRenderer().updateSourceParam(this, AudioParam.MaxDistance); 564 } 565 566 /** 567 * @return The reference playing distance for the audio node. 568 * 569 * @see AudioNode#setRefDistance(float) 570 */ getRefDistance()571 public float getRefDistance() { 572 return refDistance; 573 } 574 575 /** 576 * Set the reference playing distance for the audio node. 577 * Does nothing if the audio node is not positional. 578 * <br/> 579 * The reference playing distance is the distance at which the 580 * audio node will be exactly half of its volume. 581 * 582 * @param refDistance The reference playing distance. 583 * @throws IllegalArgumentException If refDistance is negative 584 */ setRefDistance(float refDistance)585 public void setRefDistance(float refDistance) { 586 if (refDistance < 0) { 587 throw new IllegalArgumentException("Reference distance cannot be negative"); 588 } 589 590 this.refDistance = refDistance; 591 if (channel >= 0) 592 getRenderer().updateSourceParam(this, AudioParam.RefDistance); 593 } 594 595 /** 596 * @return True if the audio node is directional 597 * 598 * @see AudioNode#setDirectional(boolean) 599 */ isDirectional()600 public boolean isDirectional() { 601 return directional; 602 } 603 604 /** 605 * Set the audio node to be directional. 606 * Does nothing if the audio node is not positional. 607 * <br/> 608 * After setting directional, you should call 609 * {@link AudioNode#setDirection(com.jme3.math.Vector3f) } 610 * to set the audio node's direction. 611 * 612 * @param directional If the audio node is directional 613 */ setDirectional(boolean directional)614 public void setDirectional(boolean directional) { 615 this.directional = directional; 616 if (channel >= 0) 617 getRenderer().updateSourceParam(this, AudioParam.IsDirectional); 618 } 619 620 /** 621 * @return The direction of this audio node. 622 * 623 * @see AudioNode#setDirection(com.jme3.math.Vector3f) 624 */ getDirection()625 public Vector3f getDirection() { 626 return direction; 627 } 628 629 /** 630 * Set the direction of this audio node. 631 * Does nothing if the audio node is not directional. 632 * 633 * @param direction 634 * @see AudioNode#setDirectional(boolean) 635 */ setDirection(Vector3f direction)636 public void setDirection(Vector3f direction) { 637 this.direction = direction; 638 if (channel >= 0) 639 getRenderer().updateSourceParam(this, AudioParam.Direction); 640 } 641 642 /** 643 * @return The directional audio node, cone inner angle. 644 * 645 * @see AudioNode#setInnerAngle(float) 646 */ getInnerAngle()647 public float getInnerAngle() { 648 return innerAngle; 649 } 650 651 /** 652 * Set the directional audio node cone inner angle. 653 * Does nothing if the audio node is not directional. 654 * 655 * @param innerAngle The cone inner angle. 656 */ setInnerAngle(float innerAngle)657 public void setInnerAngle(float innerAngle) { 658 this.innerAngle = innerAngle; 659 if (channel >= 0) 660 getRenderer().updateSourceParam(this, AudioParam.InnerAngle); 661 } 662 663 /** 664 * @return The directional audio node, cone outer angle. 665 * 666 * @see AudioNode#setOuterAngle(float) 667 */ getOuterAngle()668 public float getOuterAngle() { 669 return outerAngle; 670 } 671 672 /** 673 * Set the directional audio node cone outer angle. 674 * Does nothing if the audio node is not directional. 675 * 676 * @param outerAngle The cone outer angle. 677 */ setOuterAngle(float outerAngle)678 public void setOuterAngle(float outerAngle) { 679 this.outerAngle = outerAngle; 680 if (channel >= 0) 681 getRenderer().updateSourceParam(this, AudioParam.OuterAngle); 682 } 683 684 /** 685 * @return True if the audio node is positional. 686 * 687 * @see AudioNode#setPositional(boolean) 688 */ isPositional()689 public boolean isPositional() { 690 return positional; 691 } 692 693 /** 694 * Set the audio node as positional. 695 * The position, velocity, and distance parameters effect positional 696 * audio nodes. Set to false if the audio node should play in "headspace". 697 * 698 * @param positional True if the audio node should be positional, otherwise 699 * false if it should be headspace. 700 */ setPositional(boolean positional)701 public void setPositional(boolean positional) { 702 this.positional = positional; 703 if (channel >= 0) 704 getRenderer().updateSourceParam(this, AudioParam.IsPositional); 705 } 706 707 @Override updateGeometricState()708 public void updateGeometricState(){ 709 boolean updatePos = false; 710 if ((refreshFlags & RF_TRANSFORM) != 0){ 711 updatePos = true; 712 } 713 714 super.updateGeometricState(); 715 716 if (updatePos && channel >= 0) 717 getRenderer().updateSourceParam(this, AudioParam.Position); 718 } 719 720 @Override clone()721 public AudioNode clone(){ 722 AudioNode clone = (AudioNode) super.clone(); 723 724 clone.direction = direction.clone(); 725 clone.velocity = velocity.clone(); 726 727 return clone; 728 } 729 730 @Override write(JmeExporter ex)731 public void write(JmeExporter ex) throws IOException { 732 super.write(ex); 733 OutputCapsule oc = ex.getCapsule(this); 734 oc.write(audioKey, "audio_key", null); 735 oc.write(loop, "looping", false); 736 oc.write(volume, "volume", 1); 737 oc.write(pitch, "pitch", 1); 738 oc.write(timeOffset, "time_offset", 0); 739 oc.write(dryFilter, "dry_filter", null); 740 741 oc.write(velocity, "velocity", null); 742 oc.write(reverbEnabled, "reverb_enabled", false); 743 oc.write(reverbFilter, "reverb_filter", null); 744 oc.write(maxDistance, "max_distance", 20); 745 oc.write(refDistance, "ref_distance", 10); 746 747 oc.write(directional, "directional", false); 748 oc.write(direction, "direction", null); 749 oc.write(innerAngle, "inner_angle", 360); 750 oc.write(outerAngle, "outer_angle", 360); 751 752 oc.write(positional, "positional", false); 753 } 754 755 @Override read(JmeImporter im)756 public void read(JmeImporter im) throws IOException { 757 super.read(im); 758 InputCapsule ic = im.getCapsule(this); 759 760 // NOTE: In previous versions of jME3, audioKey was actually 761 // written with the name "key". This has been changed 762 // to "audio_key" in case Spatial's key will be written as "key". 763 if (ic.getSavableVersion(AudioNode.class) == 0){ 764 audioKey = (AudioKey) ic.readSavable("key", null); 765 }else{ 766 audioKey = (AudioKey) ic.readSavable("audio_key", null); 767 } 768 769 loop = ic.readBoolean("looping", false); 770 volume = ic.readFloat("volume", 1); 771 pitch = ic.readFloat("pitch", 1); 772 timeOffset = ic.readFloat("time_offset", 0); 773 dryFilter = (Filter) ic.readSavable("dry_filter", null); 774 775 velocity = (Vector3f) ic.readSavable("velocity", null); 776 reverbEnabled = ic.readBoolean("reverb_enabled", false); 777 reverbFilter = (Filter) ic.readSavable("reverb_filter", null); 778 maxDistance = ic.readFloat("max_distance", 20); 779 refDistance = ic.readFloat("ref_distance", 10); 780 781 directional = ic.readBoolean("directional", false); 782 direction = (Vector3f) ic.readSavable("direction", null); 783 innerAngle = ic.readFloat("inner_angle", 360); 784 outerAngle = ic.readFloat("outer_angle", 360); 785 786 positional = ic.readBoolean("positional", false); 787 788 if (audioKey != null) { 789 try { 790 data = im.getAssetManager().loadAudio(audioKey); 791 } catch (AssetNotFoundException ex){ 792 Logger.getLogger(AudioNode.class.getName()).log(Level.FINE, "Cannot locate {0} for audio node {1}", new Object[]{audioKey, key}); 793 data = PlaceholderAssets.getPlaceholderAudio(); 794 } 795 } 796 } 797 798 @Override toString()799 public String toString() { 800 String ret = getClass().getSimpleName() 801 + "[status=" + status; 802 if (volume != 1f) { 803 ret += ", vol=" + volume; 804 } 805 if (pitch != 1f) { 806 ret += ", pitch=" + pitch; 807 } 808 return ret + "]"; 809 } 810 } 811