如何在Android应用中设置手机铃声?

46

我正在尝试从我的Android活动中编写代码来设置新的默认铃声。

我已经将铃声下载到bytearray中。

13个回答

73

最终,我成功将默认铃声设置为我下载的其中一首。 以下内容只包括将其设置为默认铃声所需的代码,不包括下载代码。

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不能直接指向SD卡上的文件,例如 "/sdcard/media/ringtones/mysong.mp3"。这样不行!

你需要的是文件的外部文件URI,它可能是类似于"/external/audio/media/46"的东西。

46是MediaStore数据库中列的ID,所以你需要先将SD卡中的文件添加到数据库中。

那么,MediaStore如何维护它的ID?随着您多次执行此操作,此数字可能会变得非常高。

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


1
在这行代码中Uri newUri = main.getContentResolver().insert(uri, values);中,main是什么?我得到的新的Uri为null,但我不知道为什么。请帮帮我。 - Shailendra Madda
我在代码中使用了 getApplicationContext() 替代了 main,这样做正确吗?还有其他什么方法可以解决这个问题吗? - Shailendra Madda
在Android L中运行良好。但在Kitkat中,它会将newUri设置为null。我看到以下异常android.database.sqlite.SQLiteConstraintException: column _data is not unique。这对我添加的任何音频文件都会发生。 - Aniket Thakur
1
在 newUri 之前添加以下代码行:getContentResolver().delete(uri, MediaStore.MediaColumns.DATA + "="" + ringtoneFile.getAbsolutePath() + """, null); - AkhilGite
我想将默认铃声设置为无/静音,如何使用您的代码实现这一点。请帮忙... - Sagar
显示剩余2条评论

4

Vidar的回答太长了,并且每次想要将一首歌设置为铃声时都会添加重复条目。相反,您应该尝试以下方法:

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


                  }

1
你需要结合 Vidar 的回答和你自己的回答,在 SD 卡上已经存在文件时运行。 - dodgy_coder
如果我们有音频文件的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); - Sumit Garai

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());
  }

 }

这将帮助您从原始文件夹设置铃声,其中您放置了所有铃声。对我来说它也很好用。 - SANJAY GUPTA
soundFile = mCr.openAssetFileDescriptor(mUri, "r");请问您为什么写了这行代码? - Sritam Jagadev
这里的globalapps是什么? - Sandeep Londhe
如何将其设置为静音/无声作为默认铃声,请帮忙。 - Sagar

2
您可以使用内置的 RingtonePreference 类。 AndroidGuys在这里有一个很好的教程。

嗯,感谢提及,但 RingtonePreference 允许用户选择铃声资产供 您的 应用使用。它不会设置系统铃声。 - CommonsWare
1
好的,也许我错过了,但是参考资料并没有明确说明。我想知道除非应用程序正在处理来电,否则某人会在他的应用程序中使用什么铃声。 - russoue

1

这是我使用的代码!希望能帮到你..
这也是链接

 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);

编程愉快!


0
提供铃声选择意图。
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
            );
        }
    }

0

我无法评论该解决方案,因为我在堆栈溢出上没有足够的声望... 我只想添加一种方法将音频文件添加到媒体数据库中,而无需直接访问数据库,从而避免重复。 该解决方案基于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);
            }
        }
    );

0

这对我来说甚至在安卓S上都有效,希望它能帮助你在安卓程序中动态设置铃声。

RingtoneManager.setActualDefaultRingtoneUri(this, RingtoneManager.TYPE_RINGTONE, uri);

0
使用此函数设置铃声。
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);
        }
    }
}

音乐 ID 可以从 Cursor 中获取。希望你知道,或者在这里检查一下


0
如果被采纳的答案无法正常工作,则使用以下方法:
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" />


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接