- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在基于 this tutorial 在 Xamarin Android 中创建自己的日历.我将所需的一切从 Java 转换为 C#,但现在当我启动应用程序并打开包含行中自定义日历的 fragment 时:
public override View OnCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
var view = inflater.Inflate(Resource.Layout.calendar_fragment_main, container, false); //System.NotSupportedException Error
return view;
}
我收到一条错误消息:
System.NotSupportedException - Could not activate JNI Handle 0xffcdb7e8 (key_handle 0x741b240) of Java type 'md57f15d2d0137b5b5d70f719ce3cee21d4/EstiCalendar' as managed type 'EstiMOBILE.Droid.Components.EstiCalendar.Component.Layouts.EstiCalendar'.
还有 2 个其他错误可能来自第一个错误:
Java.Lang.UnsupportedOperationException - Binary XML file line #1: You must supply a layout_width attribute.
Android.Views.InflateException - Binary XML file line #1: Binary XML file line #1: You must supply a layout_width attribute.
包含自定义日历的 fragment (calendar_fragment_main.axml):
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:esticalendar="http://schemas.android.com/apk/res-auto"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<RelativeLayout
android:id="@+id/calendar_content_wrapper"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/calendar_title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_alignParentTop="true"
tools:text="Calendar" />
<CalendarView
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:layout_below="@id/calendar_title"
android:id="@+id/calendar_main_object"
android:visibility="gone" />
<EstiMOBILE.Droid.Components.EstiCalendar.Component.Layouts.EstiCalendar
android:layout_width="match_parent"
android:layout_height="match_parent"
android:id="@+id/esti_calendar"
esticalendar:date_format="MMMM yyyy" />
</RelativeLayout>
</RelativeLayout>
我的自定义日历类 EstiCalendar.cs
using System;
using System.Collections.Generic;
using Android.Content;
using Android.Content.Res;
using Android.Graphics;
using Android.Runtime;
using Android.Util;
using Android.Views;
using Android.Widget;
using Java.Text;
using Java.Util;
namespace EstiMOBILE.Droid.Components.EstiCalendar.Component.Layouts
{
public class EstiCalendar : LinearLayout
{
// how many days to show, defaults to six weeks, 42 days
private static readonly int DAYS_COUNT = 42;
// default date format
private static readonly string DATE_FORMAT = "MMM yyyy";
// date format
private string dateFormat;
// current displayed month
private Calendar currentDate = Calendar.Instance;
//event handling
private IEventHandler eventHandler = null;
// internal components
private LinearLayout header;
private ImageView btnPrev;
private ImageView btnNext;
private TextView txtDate;
private GridView grid;
// seasons' rainbow
int[] rainbow = new int[] {
Resource.Color.summer,
Resource.Color.fall,
Resource.Color.winter,
Resource.Color.spring
};
// month-season association (northern hemisphere, sorry australia :)
readonly int[] monthSeason = { 2, 2, 3, 3, 3, 0, 0, 0, 1, 1, 1, 2 };
public EstiCalendar(IntPtr handle, JniHandleOwnership transfer) : base(handle, transfer)
{
}
public EstiCalendar(Context context, IAttributeSet attrs) : base(context, attrs)
{
InitControl(context, attrs);
}
public EstiCalendar(Context context, IAttributeSet attrs, int defStyleAttr) : base(context, attrs, defStyleAttr)
{
InitControl(context, attrs);
}
/**
* Load control xml layout
*/
private void InitControl(Context context, IAttributeSet attrs)
{
LayoutInflater inflater = (LayoutInflater)context.GetSystemService(Context.LayoutInflaterService);
inflater.Inflate(Resource.Layout.calendar_control, this);
LoadDateFormat(attrs);
AssignUiElements();
AssignClickHandlers();
UpdateCalendar();
}
private void LoadDateFormat(IAttributeSet attrs)
{
TypedArray ta = Context.ObtainStyledAttributes(attrs, Resource.Styleable.EstiCalendar);
try
{
// try to load provided date format, and fallback to default otherwise
dateFormat = ta.GetString(Resource.Styleable.EstiCalendar_date_format);
if (dateFormat == null)
dateFormat = DATE_FORMAT;
}
finally
{
ta.Recycle();
}
}
private void AssignUiElements()
{
// layout is inflated, assign local variables to components
header = (LinearLayout)FindViewById(Resource.Id.calendar_header);
btnPrev = (ImageView)FindViewById(Resource.Id.calendar_prev_button);
btnNext = (ImageView)FindViewById(Resource.Id.calendar_next_button);
txtDate = (TextView)FindViewById(Resource.Id.calendar_date_display);
grid = (GridView)FindViewById(Resource.Id.calendar_grid);
}
private void AssignClickHandlers()
{
btnNext.Click += (sender, e) =>
{
currentDate.Add(Calendar.Month, 1);
UpdateCalendar();
};
btnPrev.Click += (sender, e) =>
{
currentDate.Add(Calendar.Month, -1);
UpdateCalendar();
};
grid.ItemLongClick += (object sender, AdapterView.ItemLongClickEventArgs e) => {
if (eventHandler != null)
{
eventHandler.OnDayLongPress((Date)e.Position);
}
};
}
public void UpdateCalendar()
{
UpdateCalendar(null);
}
/**
* Display dates correctly in grid
*/
public void UpdateCalendar(HashSet<Date> events)
{
List<Date> cells = new List<Date>();
Calendar calendar = (Calendar)currentDate.Clone();
// determine the cell for current month's beginning
calendar.Set(Calendar.DayOfMonth, 1);
int monthBeginningCell = calendar.Get(Calendar.DayOfWeek) - 1;
// move calendar backwards to the beginning of the week
calendar.Add(Calendar.DayOfMonth, -monthBeginningCell);
// fill cells
while (cells.Count < DAYS_COUNT)
{
cells.Add(calendar.Time);
calendar.Add(Calendar.DayOfMonth, 1);
}
// update grid
grid.Adapter = new CalendarAdapter(Context, cells, events);
// update title
SimpleDateFormat sdf = new SimpleDateFormat(dateFormat);
txtDate.Text = sdf.Format(currentDate.Time);
// set header color according to current season
int month = currentDate.Get(Calendar.Month);
int season = monthSeason[month];
int color = rainbow[season];
header.SetBackgroundColor(Resources.GetColor(color));
}
private class CalendarAdapter : BaseAdapter<Date>
{
// days with events
private HashSet<Date> eventDays;
private List<Date> days;
// for view inflation
private LayoutInflater inflater;
private Context context;
public CalendarAdapter(Context context, List<Date> days, HashSet<Date> eventDays) : base()
{
this.eventDays = eventDays;
this.days = days;
inflater = LayoutInflater.From(context);
this.context = context;
}
public override Date this[int position] => days[position];
public override int Count => days.Count;
public override long GetItemId(int position)
{
return position;
}
public override View GetView(int position, View convertView, ViewGroup parent)
{
// day in question
var date = GetItem(position) as Date;
int day = date.GetDate();
int month = date.Month;
int year = date.Year;
// today
Date today = new Date();
// inflate item if it does not exist yet
if (convertView == null)
convertView = inflater.Inflate(Resource.Layout.control_calendar_day, parent, false);
// if this day has an event, specify event image
convertView.SetBackgroundResource(0);
if (eventDays != null)
{
foreach (Date eventDate in eventDays)
{
if (eventDate.GetDate() == day &&
eventDate.Month == month &&
eventDate.Year == year)
{
// mark this day for event
convertView.SetBackgroundResource(Resource.Drawable.reminder);
break;
}
}
}
// clear styling
((TextView)convertView).SetTypeface(null, TypefaceStyle.Normal);
((TextView)convertView).SetTextColor(Color.Black);
if (month != today.Month || year != today.Year)
{
// if this day is outside current month, grey it out
((TextView)convertView).SetTextColor(context.Resources.GetColor(Resource.Color.greyed_out));
}
else if (day == today.GetDate())
{
// if it is today, set it to blue/bold
((TextView)convertView).SetTypeface(null, TypefaceStyle.Bold);
((TextView)convertView).SetTextColor(context.Resources.GetColor(Resource.Color.today));
}
// set text
((TextView)convertView).Text = $"{date.GetDate()}";
return convertView;
}
}
public void SetEventHandler(IEventHandler eventHandler)
{
this.eventHandler = eventHandler;
}
public interface IEventHandler
{
void OnDayLongPress(Date date);
}
}
}
CalendarFragment.cs 包含错误:
using System;
using System.Collections.Generic;
using Android.OS;
using Android.Runtime;
using Android.Views;
using Android.Widget;
using EstiMOBILE.Droid.Components.EstiCalendar.Component.Layouts;
using EstiMOBILE.Droid.Fragments.BaseFragments;
using Java.Util;
namespace EstiMOBILE.Droid.Fragments.CustomFragments
{
public class CalendarFragment : BaseFragment
{
public override void OnCreate(Bundle savedInstanceState)
{
base.OnCreate(savedInstanceState);
}
public static CalendarFragment NewInstance()
{
var frag1 = new CalendarFragment { Arguments = new Bundle() };
return frag1;
}
public override View OnCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
{
var view = inflater.Inflate(Resource.Layout.calendar_fragment_main, container, false); //Error
//InitCustomCalendar(view);
return view;
}
public void InitCustomCalendar(View view)
{
HashSet<Date> events = new HashSet<Date>
{
new Date()
};
EstiCalendar cv = (EstiCalendar) view.FindViewById(Resource.Id.esti_calendar);
cv.UpdateCalendar(events);
}
}
}
你有什么帮助,我怎样才能摆脱这个错误?
最佳答案
使用IntPtr
和JniHandleOwnership 添加构造函数
public EstiCalendar(IntPtr handle, JniHandleOwnership transfer) : base(handle, transfer)
{
}
这就是 Java 获取 Android 可调用包装器 (ACW) 的方式。所以这个 ctor 将在任何其他 ctors 之前被击中。
关于c# - 在添加自己的 LinearLayout 期间无法激活 JNI 句柄 - Xamarin Android,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54827124/
我有一个 LinearLayout,上面有几个 View 。所以我想要实现的是复制这个 LinearLayout 并在某个时候使用它。但是,当我对原始 LinearLayout 执行更改(例如方法 r
我试着做一个像这样的简单布局 仅显示第一个 TextView(“hello”)。我做错了什么? 最佳答案 制作内部布局wrap_content的l
我想构建某种 Twitter 应用程序。在 DashboardActivity 中,我需要在每次单击“发布”按钮时添加一个状态框。我的仪表板 xml 如下所示: -->header
我试图在另一个 LinearLayout 中插入一个 LinearLayout。我不知道我的做法对不对。 我需要在不使用通货膨胀的情况下尝试这种方式。 LinearLayout addre
我的 Android UI 中只有一个 ImageView 和一个 TextView。目前 TextView 与 ImageView 在同一个 LinearLayout 中,因此文本有时会跑到 2 行
我正在尝试让“播放”图标的图像显示在我在 LinearLayout 中的列表项的右侧。 但是,由于某种原因,图像出现在
我有两个 LinearLayout 结构如下。 问题是第二个 LinearLayout (id=goi) 没有按预期显示。我尝试将顶部布局更改为 Relativ
我以编程方式创建了很多 LinearLayout 并将它们放在另一个 LinearLayout 上: public void loadList(){ LinearLayout linearLa
我想创建按钮 1 到 9,并且我想循环执行此操作。但在每 3 个计数中,我想创建一个新的 LinearLayout。 final LinearLayout[] ll2 = new LinearLa
我需要创建 X 个 linearLayout - 每个 linearLayout 包含文本(项目名称)、位图(项目位图图像)。 因此,我创建了一些 for 循环来创建动态 linearLayout 和
您好,我是 Android 开发的初学者。我制作了一个 LinearLayout 以美观的方式显示不同的信息。我想要一个带有 ScrollView 的屏幕,我可以在其中添加任意数量的 LinearLa
我是 Android 编程的新手,我正在尝试设置一个包含四个组件的 LinearLayout。 目前看起来是这样的: XML:
首先,这是我的 activity_main.xml 文件的结构: 相对布局 线性布局 fragment (谷歌地图) 线性布局 搜索栏 TextView 图片按钮 我试图将第二个线性布局(不包含 ma
我正在创建一个自定义复合布局,它由一个可点击的水平 LinearLayout 组成,其中包含一个 ImageView 和两个 TextView。最终,我希望能够使用单个字段引用整个内容,并根据用户 A
我在 main.xml 中有一个 LinearLayout: 我制作了另一个 XML 文件,名为 item_box.xml: 基本上,我想通过代码(以编程方式
我有一个 LinearLayout,它应该包含一个项目列表(每个项目都是一个 LinearLayout)。 问题是子项目 (LinearLayouts) 没有显示在另一个下面:只有第一个是可见的。 这
下午好 我正在尝试像这样进行倒计时: 为此,我制作了两个水平的 LinearLayout,一个用于数值,另一个放置在下方,用于标签,例如“DAYS”或“HOURS”。 我的目标是在数值中间对齐标签,如
我得到了这个应用程序(顺便说一句,它和完成的一样好),我需要在其中动态地将 LinearLayout View 添加到 LinearLayout 容器。它的工作方式是用户在 EditText 中写一些
在 Udacity Android 初学者类(class)之后,我想向 list_item View 添加一个音频图标按钮。现在 list_item.xml 有一个父级水平线性布局和一个嵌套的垂直线性
我得到了一个相对布局,其中包含另一个我用来替换“标题”的相对布局,一个稍后用作“控制面板”的线性布局,以及一个 horizontalScrollView,其中 horizontalScroll
我是一名优秀的程序员,十分优秀!