如何通过我的活动在 Android 中设置铃声?

Posted

技术标签:

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

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

我已经将铃声下载到bytearray

【问题讨论】:

【参考方案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());

这适用于我的。

【讨论】:

以上是关于如何通过我的活动在 Android 中设置铃声?的主要内容,如果未能解决你的问题,请参考以下文章

如何在Android中设置铃声+震动

如何在 UILocalNotification 中设置用户铃声的声音?

在android中的不同活动中设置按钮上的时间

如何在每天不同的时间在android studio中设置每日重复通知?

如何在选项卡式活动中设置片段从右到左开始

在 Android 中设置 DatePickerDialog 的限制?