Файл Ics не анализирует и не получает данные, как ожидалось в Android. (используя библиотеку ical4) - PullRequest
1 голос
/ 30 октября 2019

Я пытаюсь разобрать файл ics с помощью библиотеки примеров кода: Вот ссылка на пример проекта, который я использую / ссылаюсь на: ссылка на проект

Как только я пытаюсьПри разборе данных происходит сбой с исключением из-за нулевого указателя или выдается сообщение о невозможности анализа файла. (Я добавил весь код для проекта в моей ссылке выше, довольно маленький проект). Вот основной код активности, который я использую для этого:

public class MainActivity extends AppCompatActivity {

    public static final String TAG = MainActivity.class.getCanonicalName();

    public static final Boolean DEBUG = true; //BuildConfig.DEBUG;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Intent i = getIntent();
        Log.d(TAG, "Dumping Intent start");
        Log.d(TAG, "Action: "+i.getAction());
        Log.d(TAG, "URI: "+i.getDataString());
        Log.d(TAG, "Type: "+i.getType());
        Bundle bundle = i.getExtras();
        if (bundle != null) {
            Set<String> keys = bundle.keySet();
            Iterator<String> it = keys.iterator();
            while (it.hasNext()) {
                String key = it.next();
                Log.d(TAG, "[" + key + "=" + bundle.get(key) + "]");
            }
        }
        Log.d(TAG, "Dumping Intent end");

        Log.d(TAG, getFilePathFromUri(this, i.getData()));
        try {
            startActivity(new ICALParser(new FileInputStream(getFilePathFromUri(this, i.getData()))).buildIntent());
        } catch (Exception e) {
            if(DEBUG) Log.e(TAG, "Couldn't parse",e);
        }

        if(DEBUG) Log.d(TAG, "Done");
        finish();

    }

    //http://stackoverflow.com/a/23750660
    public static String getFilePathFromUri(Context c, Uri uri) {
        String filePath = new File(uri.getPath()).getAbsolutePath();
        if ("content".equals(uri.getScheme())) {
            String[] filePathColumn = { MediaStore.MediaColumns.DATA };
            ContentResolver contentResolver = c.getContentResolver();
            Cursor cursor = contentResolver.query(uri, filePathColumn, null, null, null);
            cursor.moveToFirst();
            int columnIndex = cursor.getColumnIndex(filePathColumn[0]);
             filePath = cursor.getString(columnIndex);
            cursor.close();
        } else if ("file".equals(uri.getScheme())) {
             filePath = new File(uri.getPath()).getAbsolutePath();
        }
        return filePath;
    }
}

Кажется, что проблема возникает в: getFilePathFromUri, где он либо аварийно завершает работу с исключением нулевого указателя как: android.net.Uri.getScheme (), либо генерирует случайныйошибка разбора, которая каждый раз отличается по своему содержанию. Вот класс парсера, который я использую:

public class ICALParser {
    public static final String TAG = ICALParser.class.getCanonicalName();
    public static final boolean DEBUG = MainActivity.DEBUG;

    private Calendar calendar;
    private VEvent event;

    public ICALParser(InputStream ics) throws IOException, ParserException {
        CalendarBuilder builder = new CalendarBuilder();
        calendar = builder.build(ics);
        event = findFirstEvent();
    }

    public Intent buildIntent(){
        Intent i = new Intent(Intent.ACTION_INSERT, CalendarContract.Events.CONTENT_URI);

        if(DEBUG) {
            for (Object o : event.getProperties()) {
                Property property = (Property) o;
                Log.d(TAG, "Property [" + property.getName() + ", " + property.getValue() + "]");
            }
        }

        i.putExtra(CalendarContract.Events.TITLE, getValueOrNull(Property.SUMMARY));
        i.putExtra(CalendarContract.Events.DESCRIPTION, getValueOrNull(Property.DESCRIPTION));
        i.putExtra(CalendarContract.Events.EVENT_LOCATION, getValueOrNull(Property.LOCATION));

        long start = event.getStartDate().getDate().getTime();
        long end = event.getEndDate().getDate().getTime();
        i.putExtra(CalendarContract.EXTRA_EVENT_BEGIN_TIME,start);
        i.putExtra(CalendarContract.EXTRA_EVENT_END_TIME, end);

        boolean allDayEvent = ((event.getProperty("X-MICROSOFT-CDO-ALLDAYEVENT") != null            //Microsoft's custom field exists
                && event.getProperty("X-MICROSOFT-CDO-ALLDAYEVENT").getValue().equals("true"))  //  and is true, or
                || (end-start % 1000*60*60*24 == 0 ) );                                             //the duration is an integer number of days
        i.putExtra(CalendarContract.EXTRA_EVENT_ALL_DAY, allDayEvent);

        i.putExtra(CalendarContract.Events.RRULE, getValueOrNull(Property.RRULE));

        return i;
    }

    private String getValueOrNull(String name){
        Property p = event.getProperty(name);
        return p == null ? null : p.getValue();
    }

    private VEvent findFirstEvent(){
        for (Object o : calendar.getComponents()) {
            Component c = (Component) o;
            VEvent e = c instanceof VEvent ? ((VEvent) c) : null;
            if (e != null) {
                return e;
            }
        }
        return null;
    }


}

Прикрепленный выше проект компилируется и запускается, однако файлы ics, вложенные в папку raw, не анализируются и не отображаются в календари на моем устройстве. Любая идея, как я могу решить эту проблему?

Спасибо!

1 Ответ

1 голос
/ 06 ноября 2019

Это кажется излишним. Почему бы не добавить событие непосредственно в календарь с помощью следующего кода? :

Intent intent = new Intent(Intent.ACTION_INSERT);
        intent.setData(CalendarContract.Events.CONTENT_URI);
// pass the values below somehow , either through a callback or some method, whatever you're using. 
        intent.putExtra(CalendarContract.Events.TITLE, contentTitle);
        intent.putExtra(CalendarContract.Events.EVENT_LOCATION, locationAddress);
        intent.putExtra(CalendarContract.Events.DESCRIPTION, contentDescription);

        intent.putExtra(CalendarContract.EXTRA_EVENT_BEGIN_TIME, startTimeInMilliseconds);
        intent.putExtra(CalendarContract.EXTRA_EVENT_END_TIME, endTimeInMilliseconds );
            intent.putExtra(CalendarContract.Events.EVENT_TIMEZONE, eventTimeZone);
            intent.putExtra(CalendarContract.Events.EVENT_END_TIMEZONE, eventTimeZone); // pass your event time zone here, if any.

        // Making it private and shown as busy
        intent.putExtra(CalendarContract.Events.ACCESS_LEVEL, CalendarContract.Events.ACCESS_PRIVATE);
        intent.putExtra(CalendarContract.Events.AVAILABILITY, CalendarContract.Events.AVAILABILITY_BUSY);
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

        startActivity(intent);
...