2013-02-26 51 views
0

我有一个片段,它包含我的应用程序的所有配置文件的列表。该列表使用自定义BaseAdapter。以下是此适配器的规格:开关自定义BaseAdapter问题

  • 用户一次只能启用一个配置文件。
  • 如果用户尝试禁用所有配置文件,则用户将被警告并且上次启用的配置文件将被设置为启用。
  • 如果用户尝试启用未配置的配置文件,将会警告用户,并且上次启用的配置文件将设置为启用。
  • 用户可以配置每个配置文件。

一切工作发现除,这里是我目前的执行存在的第一次创建列表中的一个问题:

  • 如果用户启用(用户检查开关)在第一轮廓列表第一次创建列表,然后尝试启用另一个配置文件,第一个仍然被选中(Switch仍然被选中)。另请注意,即使选中了第一个配置文件,实际启用的配置文件也将是稍后检查的配置文件。
  • 如果选择其他配置文件,则不会发生这种情况。每当启用其他 配置文件时,所有其他配置文件都被禁用。
  • 但是,活动暂停并恢复后,一切工作 像预期的那样。意思是第一个开关仍然 获得检查问题将消失。

下面是此问题的图形表示: enter image description here

下面是该片段承载BaseAdapter(AdapterProfiles)的代码。 PreferenceData是处理与共享偏好:

public class Profiles extends Fragment implements OnCustomClickListener { 

private View view_profiles; 
private ListView lv_profiles; 

private PreferenceData dataPreferenceObj; 

private AdapterProfiles adapter; 
private ArrayList<String> profileList = new ArrayList<String>(); 
private final ArrayList<HashMap<String, String>> profilesList = new ArrayList<HashMap<String, String>>(); 

public View onCreateView(LayoutInflater inflater, ViewGroup container, 
     Bundle savedInstanceState) { 
    // ... 
    lv_profiles = (ListView) view_profiles.findViewById(R.id.lv_profiles); 
    // ... 
} 

@Override 
public void onResume() { 
    createProfileList(); 
    // Registers broadcast receiver 

    super.onResume(); 
} 

private void createProfileList() { 
    if (profilesList.isEmpty() == true) { 
     profileList = new ProfileList().getProfileList(); 

     for (int i = 0; i < profileList.size(); i++) { 
      HashMap<String, String> map = new HashMap<String, String>(); 
      map.put(Keys.KEY_PROFILE.toString(), profileList.get(i)); 
      profilesList.add(map); 
     } 
    } 

    adapter = new AdapterProfiles(getActivity(), profilesList, this); 
    lv_profiles.setAdapter(adapter); 
} 

/* 
* This is a custom onclick listener that gets handle from of list from 
* LazyAdapterProfiles 
*/ 
@Override 
public void OnCustomClick(View view, int position) { 
    switch (view.getId()) { 
    case R.id.iv_profiles_settings: 
     // Opens setting for settings of profile clicked on 
     break; 
    case R.id.sw_profiles: 
     Switch sw_profiles = (Switch) view.findViewById(R.id.sw_profiles); 

     // Stops user from enabling profile that is not configured. 
     if (dataPreferenceObj.getTemporaryRingtoneUri().equals("") == true 
       || dataPreferenceObj.getTemporaryLocation().equals("") == true) { 
      sw_profiles.setChecked(false); 

      showWarningDialog(
        "Profile Disabled", 
        "Profile location and ringtone must be selected in order to enable this profile."); 
     } else if (dataPreferenceObj.getTemporaryRingtoneUri().equals("") != true 
       && dataPreferenceObj.getTemporaryLocation().equals("") != true) { 
      if (sw_profiles.isChecked() == true) { 
       unselectProfileState(position); 
      } else if (sw_profiles.isChecked() == false) { 
       // Stops user from disabling profile as atleast one profile 
       // must be enabled 
       showWarningDialog("Profile Cannot Disable", 
         "Atleast one profile must be enabled"); 

       sw_profiles.setChecked(true); 
      } 

      // Rest of the code that writes the preference and broadcast it 
     } 
     break; 
    } 
} 

// Gets switch map of all profiles and uncheck profile other than one that 
// is selected. 
private void unselectProfileState(int position) { 
    for (int i = 0; i < profileList.size(); i++) { 
     if (position != i) { 
      AdapterProfiles.getMapHolder().get(i).sw_profiles 
        .setChecked(false); 
     } 
    } 
} 

}

我有一个自定义BaseAdapter及其各行持有的TextView,ImageView的和开关,一切类看起来像下面: enter image description here

这是我实现的是,在列表中创建行BaseAdapter的:

public class AdapterProfiles extends BaseAdapter { 

private ArrayList<HashMap<String, String>> data; 
private static LayoutInflater inflater = null; 

private String profile_name; 
private String retrived_profile_name; 

private ViewHolder holder; 
private Activity activity; 

private OnCustomClickListener callback; 
private static SparseArray<ViewHolder> mapHolder = new SparseArray<ViewHolder>(); 

public AdapterProfiles(Activity activity, 
     ArrayList<HashMap<String, String>> data, 
     OnCustomClickListener callback) { 

    this.activity = activity; 
    this.data = data; 
    this.callback = callback; 

    inflater = (LayoutInflater) activity 
      .getSystemService(Context.LAYOUT_INFLATER_SERVICE); 
} 

public AdapterProfiles(Activity activity, 
     ArrayList<HashMap<String, String>> data) { 
    this.data = data; 

    inflater = (LayoutInflater) activity 
      .getSystemService(Context.LAYOUT_INFLATER_SERVICE); 
} 

public View getView(int position, View convertView, ViewGroup parent) { 

    if (convertView == null) { 
     convertView = inflater.inflate(R.layout.ui_profiles_row, null); 

     holder = new ViewHolder(); 

     holder.tv_profiles_name = (TextView) convertView 
       .findViewById(R.id.tv_profiles_name); 
     holder.tv_profiles_location_name = (TextView) convertView 
       .findViewById(R.id.tv_profiles_location_name); 
     holder.iv_profiles_settings = (ImageView) convertView 
       .findViewById(R.id.iv_profiles_settings); 
     holder.sw_profiles = (Switch) convertView 
       .findViewById(R.id.sw_profiles); 

     holder.iv_profiles_settings 
       .setOnClickListener(new CustomOnClickListener(callback, 
         position)); 
     holder.sw_profiles.setOnClickListener(new CustomOnClickListener(
       callback, position)); 

     convertView.setTag(holder); 

     mapHolder.put(position, holder); 
    } else { 
     holder = (ViewHolder) convertView.getTag(); 
    } 

    getPrefs(position); 
    setUIData(); 

    return convertView; 
} 

private void getPrefs(int position) { 
    HashMap<String, String> word = new HashMap<String, String>(); 
    word = data.get(position); 

    profile_name = word.get(Keys.KEY_PROFILE.toString()); 

    SharedPreferences settings = activity.getSharedPreferences(
      Keys.PREFS_APP.toString(), 0); 
    // This allows us to retrive unique data based on the profile names 
    String PROFILE_LOCATION = Keys.KEY_LOCATION.toString() + "_" 
      + profile_name; 
    retrived_profile_name = settings.getString(Keys.KEY_PROFILE.toString(), 
      ""); 
} 

// Sets UI data of item in the list 
private void setUIData() { 
    if (profile_name.equals(retrived_profile_name) == true) { 
     holder.sw_profiles.setChecked(true); 
    } else if (profile_name.equals(retrived_profile_name) == false) { 
     holder.sw_profiles.setChecked(false); 
    } 
} 

public static class ViewHolder { 
    public ImageView iv_profiles_settings; 
    public TextView tv_profiles_location_name; 
    public TextView tv_profiles_name; 
    public Switch sw_profiles; 
} 

public static SparseArray<ViewHolder> getMapHolder() { 
    return mapHolder; 
} 

}

终于这里是CustomOnClickListener类,如果有人想知道它是什么。这个类的实现是将回调映射到列表中的特定组件。这个类最初是由某人创建的(我没有在我的书签中找到它来称赞作者),我使用他/她的示例来适合我自己的实现。下面是该代码:

public class CustomOnClickListener implements OnClickListener { 
private int position; 
private OnCustomClickListener callback; 

public CustomOnClickListener(OnCustomClickListener callback, int position) { 
    this.position = position; 
    this.callback = callback; 
} 

@Override 
public void onClick(View v) { 
    callback.OnCustomClick(v, position); 
} 

} 

public interface OnCustomClickListener { 
    public void OnCustomClick(View view, int position); 
} 

回答

0

现在,这似乎是在这里实现我自己的方法来取消选择所有其他交换机-ES在适配器的时候我知道,notifyDataSetChanged()方法将刷新视图改变愚蠢的。现在,我似乎被删除这些代码已经解决了这个问题:

if (sw_profiles.isChecked() == true) { 
       unselectProfileState(position); 
      } 

private void unselectProfileState(int position) { 
for (int i = 0; i < profileList.size(); i++) { 
    if (position != i) { 
     AdapterProfiles.getMapHolder().get(i).sw_profiles 
       .setChecked(false); 
    } 
} 

}

而且在OnCustomClick情况下R.id.sw_profiles的末尾添加以下。如前所述,这里是Android文档对此方法的快速描述“通知附加的观察者,底层数据已更改,并且任何反映数据集的视图都应刷新自身。”

adapter.notifyDataSetChanged();