【问题标题】:How to set ringtone in Android from my activity?如何通过我的活动在 Android 中设置铃声?
【发布时间】:2010-11-19 07:07:51
【问题描述】:

我正在尝试找到一种方法,通过我的 Android 活动中的代码设置新的默认铃声。

我已经将铃声下载到bytearray

【问题讨论】:

    标签: android android-activity ringtone


    【解决方案1】:

    最后,我设法将默认铃声设置为我下载的铃声。 下面不包含下载代码,仅包含将其设置为默认铃声所需的代码。

    File k = new File(path, "mysong.mp3"); // path is a file to /sdcard/media/ringtone
    
    ContentValues values = new ContentValues();
    values.put(MediaStore.MediaColumns.DATA, k.getAbsolutePath());
    values.put(MediaStore.MediaColumns.TITLE, "My Song title");
    values.put(MediaStore.MediaColumns.SIZE, 215454);
    values.put(MediaStore.MediaColumns.MIME_TYPE, "audio/mp3");
    values.put(MediaStore.Audio.Media.ARTIST, "Madonna");
    values.put(MediaStore.Audio.Media.DURATION, 230);
    values.put(MediaStore.Audio.Media.IS_RINGTONE, true);
    values.put(MediaStore.Audio.Media.IS_NOTIFICATION, false);
    values.put(MediaStore.Audio.Media.IS_ALARM, false);
    values.put(MediaStore.Audio.Media.IS_MUSIC, false);
    
    //Insert it into the database
    Uri uri = MediaStore.Audio.Media.getContentUriForPath(k.getAbsolutePath());
    Uri newUri = this.getContentResolver().insert(uri, values);
    
    RingtoneManager.setActualDefaultRingtoneUri(
      myActivity,
      RingtoneManager.TYPE_RINGTONE,
      newUri
    );  
    

    无论如何,我并不完全理解这段代码在做什么。

    铃声管理器需要一个指向要设置为新铃声的文件的 uri。但是这个uri不能像“/sdcard/media/ringtones/mysong.mp3”一样直接到sdcard。那不行!

    您需要的是文件的外部文件 uri,可能类似于 "/external/audio/media/46"

    46是MediaStore数据库中列的id,所以需要先将sdcard文件添加到数据库中。

    无论如何,mediastore 是如何维护它的 id 的呢?当您多次执行此操作时,此数字可能会变得非常高。

    我需要自己删除这一行吗?问题是有时我什至无法控制文件的删除,因为它可以使用文件浏览器直接从 sdcard 中删除。

    【讨论】:

    • 与上面代码中的媒体内容提供者交互每次都会创建一个新条目。您可能想要跟踪您正在生成的 URI,这样您就可以避免重新下载,并直接跳到 RingtoneManager
    • what is main in this line Uri newUri = main.getContentResolver().insert(uri, values); 我得到新的 Uri 为空,为什么我没有。请帮帮我..
    • 我用getApplicationContext() 代替main 是正确的还是我可以用什么来解决这个问题..
    • 在 Android L 中工作正常。但它奇巧它给newUri 为空。我看到以下异常android.database.sqlite.SQLiteConstraintException: column _data is not unique。我添加的任何音频文件都会发生这种情况。
    • 在 newUri getContentResolver().delete(uri, MediaStore.MediaColumns.DATA + "=\"" + ringtoneFile.getAbsolutePath() + "\"", null)之前添加下面的代码行;
    【解决方案2】:

    Answer By Vidar 太长,每次您想将歌曲设置为铃声时都会添加重复条目。相反,你应该试试这个

    Uri newUri=Uri.parse("content://media/external/audio/media/"+ID);  
    try {
          RingtoneManager.setActualDefaultRingtoneUri(context, RingtoneManager.TYPE_RINGTONE, newUri);
        }
    catch (Throwable t) {
    
    
                      }
    

    【讨论】:

    • 您需要将 Vidar 的答案和您的答案结合起来,当文件已经存在于 SD 卡上时,应该运行它。
    • 如果我们有音频文件的id,我们可以用这个代替,Uri uri; if(path_to_the_file.startsWith("/system")){ uri = Uri.parse(MediaStore.Audio.Media.INTERNAL_CONTENT_URI.toString() +"/"+ id); } else uri = Uri.parse(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI.toString() +"/"+ id);
    【解决方案3】:
    public void setRingtone() {
      String ringtoneuri = Environment.getExternalStorageDirectory().getAbsolutePath() + "/media/ringtone";
      File file1 = new File(ringtoneuri);
      file1.mkdirs();
      File newSoundFile = new File(ringtoneuri, "myringtone.mp3");
    
    
      Uri mUri = Uri.parse("android.resource://globalapps.funnyringtones/raw/sound_two.mp3");
    
    
      ContentResolver mCr = this.getContentResolver();
      AssetFileDescriptor soundFile;
      try {
       soundFile = mCr.openAssetFileDescriptor(mUri, "r");
      } catch (FileNotFoundException e) {
       soundFile = null;
      }
    
      try {
       byte[] readData = new byte[1024];
       FileInputStream fis = soundFile.createInputStream();
       FileOutputStream fos = new FileOutputStream(newSoundFile);
       int i = fis.read(readData);
    
       while (i != -1) {
        fos.write(readData, 0, i);
        i = fis.read(readData);
       }
    
       fos.close();
      } catch (IOException io) {
      }
    
      ContentValues values = new ContentValues();
      values.put(MediaStore.MediaColumns.DATA, newSoundFile.getAbsolutePath());
      values.put(MediaStore.MediaColumns.TITLE, "my ringtone");
      values.put(MediaStore.MediaColumns.MIME_TYPE, "audio/mp3");
      values.put(MediaStore.MediaColumns.SIZE, newSoundFile.length());
      values.put(MediaStore.Audio.Media.ARTIST, R.string.app_name);
      values.put(MediaStore.Audio.Media.IS_RINGTONE, true);
      values.put(MediaStore.Audio.Media.IS_NOTIFICATION, true);
      values.put(MediaStore.Audio.Media.IS_ALARM, true);
      values.put(MediaStore.Audio.Media.IS_MUSIC, false);
    
      Uri uri = MediaStore.Audio.Media.getContentUriForPath(newSoundFile.getAbsolutePath());
      Uri newUri = mCr.insert(uri, values);
      try {
       Uri rUri = RingtoneManager.getValidRingtoneUri(this);
       if (rUri != null)
        ringtoneManager.setStopPreviousRingtone(true);
       RingtoneManager.setActualDefaultRingtoneUri(getApplicationContext(), RingtoneManager.TYPE_RINGTONE, newUri);
       Toast.makeText(this, "New Rigntone set", Toast.LENGTH_SHORT).show();
      } catch (Throwable t) {
       Log.e("sanjay in catch", "catch exception"+e.getMessage());
      }
    
     }
    

    【讨论】:

    • 这将帮助您从放置所有铃声的原始文件夹中设置铃声。这对我来说也很好
    • soundFile = mCr.openAssetFileDescriptor(mUri, "r");你能告诉我你写这行的原因吗?
    • 这里的 globalapps 是什么?
    • 如何将其设置为静音/无作为默认铃声,请帮助
    【解决方案4】:

    您可以使用内置的 RingtonePreference 类。 AndroidGuys 对此here 有一个很好的教程。

    【讨论】:

    • 嗯,感谢您的支持,但 RingtonePreference 允许用户为您的应用程序选择铃声资产。它不设置系统铃声。
    • 好吧,我可能错过了,但参考文献没有明确说明这一点。我想知道除非应用程序正在处理来电,否则有人会在他的应用程序中处理铃声。
    【解决方案5】:

    这是我使用的代码!我希望它有帮助..
    这也是link

     String exStoragePath =    Environment.getExternalStorageDirectory().getAbsolutePath();
    String path=(exStoragePath +"/media/alarms/"); 
    
    saveas(RingtoneManager.TYPE_RINGTONE); 
    
    sendBroadcast(new Intent(Intent.ACTION_MEDIA_MOUNTED,       Uri.parse("file://"+path+filename+".mp3"
      + Environment.getExternalStorageDirectory()))); 
    
    
     File k = new File(path, filename);
    
    ContentValues values = new ContentValues(4);   
    long current = System.currentTimeMillis();
    values.put(MediaStore.MediaColumns.DATA, path + filename  );
    values.put(MediaStore.MediaColumns.TITLE,  filename ); 
    values.put(MediaStore.Audio.Media.DATE_ADDED, (int) (current / 1000));
    values.put(MediaStore.Audio.Media.MIME_TYPE, "audio/3gpp");
    
    //new
     values.put(MediaStore.Audio.Media.ARTIST, "cssounds ");
    values.put(MediaStore.Audio.Media.IS_RINGTONE, true);
    values.put(MediaStore.Audio.Media.IS_NOTIFICATION, false);
    values.put(MediaStore.Audio.Media.IS_ALARM, true);
    values.put(MediaStore.Audio.Media.IS_MUSIC, false);  
    
       // Insert it into the database
    this.getContentResolver()
       .insert(MediaStore.Audio.Media.getContentUriForPath(k
    .getAbsolutePath()), values);
    

    编码愉快!

    【讨论】:

      【解决方案6】:

      我无法评论解决方案,因为我在堆栈溢出方面没有足够的声誉......我只想添加一种将音频文件添加到媒体数据库中的方法,而无需直接访问数据库,从而避免重复。 解决方案是基于MediaScannerConnection,这是我使用的代码:

          String[] files = { audioFullPath };
          MediaScannerConnection.scanFile(
              getApplicationContext(),
              files,
              null,
              new OnScanCompletedListener() {
                  @Override
                  public void onScanCompleted(String path, Uri uri) {
                      Log.v("myapp", "file " + path + " was scanned seccessfully: " + uri);
                  }
              }
          );
      

      【讨论】:

        【解决方案7】:

        提供铃声选择的意图。

        final Uri currentTone= RingtoneManager.getActualDefaultRingtoneUri(MainActivity.this, RingtoneManager.TYPE_ALARM);
                        Intent intent = new Intent(RingtoneManager.ACTION_RINGTONE_PICKER);
                        intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TYPE, RingtoneManager.TYPE_RINGTONE);
                        intent.putExtra(RingtoneManager.EXTRA_RINGTONE_TITLE, "Select Tone");
                        intent.putExtra(RingtoneManager.EXTRA_RINGTONE_EXISTING_URI, currentTone);
                        intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_SILENT, false);
                        intent.putExtra(RingtoneManager.EXTRA_RINGTONE_SHOW_DEFAULT, true);
                        startActivityForResult(intent, 999);
        

        然后在onActivityResult中捕捉选择的结果。

        protected void onActivityResult(int requestCode, int resultCode, Intent data) {
                if(requestCode == 999 && resultCode == RESULT_OK){
                    Uri uri = data.getParcelableExtra(RingtoneManager.EXTRA_RINGTONE_PICKED_URI);
                    txtView.setText("From :" + uri.getPath());
                    //Set selected ringtone here.
                    RingtoneManager.setActualDefaultRingtoneUri(
                            this,
                            RingtoneManager.TYPE_RINGTONE,
                            uri
                    );
                }
            }
        

        【讨论】:

          【解决方案8】:

          我已经尝试了这些代码它的帮助

            private void setRingtone(Context context, String path) {
              if (path == null) {
                  return;
              }
              File file = new File(path);
              ContentValues contentValues = new ContentValues();
              contentValues.put(MediaStore.MediaColumns.DATA, file.getAbsolutePath());
              String filterName = path.substring(path.lastIndexOf("/") + 1);
              contentValues.put(MediaStore.MediaColumns.TITLE, filterName);
              contentValues.put(MediaStore.MediaColumns.MIME_TYPE, "audio/mp3");
              contentValues.put(MediaStore.MediaColumns.SIZE, file.length());
              contentValues.put(MediaStore.Audio.Media.IS_RINGTONE, true);
              Uri uri = MediaStore.Audio.Media.getContentUriForPath(path);
              Cursor cursor = context.getContentResolver().query(uri, null, MediaStore.MediaColumns.DATA + "=?", new String[]{path}, null);
              if (cursor != null && cursor.moveToFirst() && cursor.getCount() > 0) {
                  String id = cursor.getString(0);
                  contentValues.put(MediaStore.Audio.Media.IS_RINGTONE, true);
                  context.getContentResolver().update(uri, contentValues, MediaStore.MediaColumns.DATA + "=?", new String[]{path});
                  Uri newuri = ContentUris.withAppendedId(uri, Long.valueOf(id));
                  try {
                      RingtoneManager.setActualDefaultRingtoneUri(context, RingtoneManager.TYPE_RINGTONE, newuri);
                      Toast.makeText(context, "Set as Ringtone Successfully.", Toast.LENGTH_SHORT).show();
                  } catch (Throwable t) {
                      t.printStackTrace();
                  }
                  cursor.close();
              }
          }
          

          【讨论】:

            【解决方案9】:

            如果接受的答案不起作用,请使用:

            MediaStore.Audio.Media.INTERNAL_CONTENT_URI
            

            而不是这个:

            MediaStore.Audio.Media.getContentUriForPath()
            

            在向数据库中插入值时。

            例如:

            // Defining ringtone.....
            ContentValues values = new ContentValues();
            values.put(MediaStore.MediaColumns.DATA, file.getAbsolutePath());
            values.put(MediaStore.MediaColumns.TITLE, "Sonify");
            values.put(MediaStore.MediaColumns.MIME_TYPE, "audio/mp3");
            values.put(MediaStore.Audio.Media.IS_RINGTONE, true);
            values.put(MediaStore.Audio.Media.IS_NOTIFICATION,false);
            values.put(MediaStore.Audio.Media.IS_ALARM, false);
            values.put(MediaStore.Audio.Media.IS_MUSIC, false);
            
            // Setting ringtone....
            getContentResolver().delete(MediaStore.Audio.Media.INTERNAL_CONTENT_URI,MediaStore.Audio.Media.TITLE + " = \"Sonify\"",null); 
            // To avoid duplicate inserts
            Uri ringUri = getContentResolver().insert(MediaStore.Audio.Media.INTERNAL_CONTENT_URI, values);
            RingtoneManager.setActualDefaultRingtoneUri(this, RingtoneManager.TYPE_ALARM, ringUri);
            

            别忘了添加

            <uses-permission android:name="android.permission.WRITE_SETTINGS" />

            【讨论】:

              【解决方案10】:

              使用此功能设置铃声

              private void setAsRingtone(String musicId) {
                  if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
                      if (Settings.System.canWrite(this)) {
                          Uri uri = ContentUris.withAppendedId(MediaStore.Audio.Media.EXTERNAL_CONTENT_URI, Long.parseLong(musicId));
                          RingtoneManager.setActualDefaultRingtoneUri(
                                  this,
                                  RingtoneManager.TYPE_RINGTONE,
                                  uri
                          );
                          Toast.makeText(this, "Ring set successfully", Toast.LENGTH_SHORT).show();
                      } else {
                          Intent intent = new Intent(android.provider.Settings.ACTION_MANAGE_WRITE_SETTINGS);
                          intent.setData(Uri.parse("package:" + getPackageName()));
                          intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
                          startActivity(intent);
                      }
                  }
              }
              

              Music Id 可以从 Cursor 获得。希望你知道,或者 Check it here

              【讨论】:

                【解决方案11】:

                我从 Android 的媒体应用程序中找到了这段代码。

                Settings.System.putString(resolver, 
                Settings.System.RINGTONE, ringUri.toString());
                

                这适用于我的。

                【讨论】:

                  猜你喜欢
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  • 2016-04-04
                  • 1970-01-01
                  • 1970-01-01
                  • 1970-01-01
                  相关资源
                  最近更新 更多