Last active
June 27, 2019 13:26
-
-
Save StelianMorariu/8180302 to your computer and use it in GitHub Desktop.
Seek bar preference
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<?xml version="1.0" encoding="UTF-8"?> | |
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" | |
android:layout_width="match_parent" | |
android:layout_height="wrap_content" | |
android:padding="8dp"> | |
<TextView | |
android:id="@+id/seekbarMinLabel" | |
android:layout_width="wrap_content" | |
android:layout_height="wrap_content" | |
android:layout_alignParentLeft="true" | |
android:layout_centerVertical="true" | |
android:gravity="right" | |
android:text="0" | |
android:textSize="16sp" /> | |
<TextView | |
android:id="@+id/seekbarMaxLabel" | |
android:layout_width="wrap_content" | |
android:layout_height="wrap_content" | |
android:layout_alignParentRight="true" | |
android:layout_centerVertical="true" | |
android:text="255" | |
android:textSize="16sp" /> | |
<LinearLayout | |
android:id="@+id/seekBarPrefBarContainer" | |
android:layout_width="match_parent" | |
android:layout_height="wrap_content" | |
android:layout_centerVertical="true" | |
android:layout_toLeftOf="@id/seekbarMaxLabel" | |
android:layout_toRightOf="@id/seekbarMinLabel" | |
android:orientation="horizontal" | |
android:layout_marginLeft="4dp" | |
android:layout_marginRight="4dp"> | |
<SeekBar | |
android:id="@+id/seekBarPrefSeekBar" | |
android:layout_width="match_parent" | |
android:layout_height="wrap_content" /> | |
</LinearLayout> | |
</RelativeLayout> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package ro.stemo.android.clockwidget.preferences; | |
import android.content.Context; | |
import android.content.res.TypedArray; | |
import android.preference.Preference; | |
import android.util.AttributeSet; | |
import android.util.Log; | |
import android.view.View; | |
import android.view.ViewGroup; | |
import android.view.ViewParent; | |
import android.widget.LinearLayout; | |
import android.widget.SeekBar; | |
import android.widget.TextView; | |
import ro.stemo.android.clockwidget.R; | |
/** | |
* Created by Stelian Morariu on 12/9/13. | |
*/ | |
public class SeekBarPreference extends Preference implements SeekBar.OnSeekBarChangeListener { | |
private static final String ANDROIDNS = "http://schemas.android.com/apk/res/android"; | |
private static final String APPLICATIONNS = "http://stemo.ro"; | |
private static final int DEFAULT_VALUE = 255; | |
private static final String LOG_TAG = SeekBarPreference.class.getSimpleName(); | |
private int mMaxValue = 255; | |
private int mMinValue = 0; | |
private int mInterval = 1; | |
private int mCurrentValue; | |
private SeekBar mSeekBar; | |
private TextView mStatusText; | |
public SeekBarPreference(Context context, AttributeSet attrs) { | |
super(context, attrs); | |
initPreference(context, attrs); | |
} | |
public SeekBarPreference(Context context, AttributeSet attrs, int defStyle) { | |
super(context, attrs, defStyle); | |
initPreference(context, attrs); | |
} | |
private void initPreference(Context context, AttributeSet attrs) { | |
setValuesFromXml(attrs); | |
mSeekBar = new SeekBar(context, attrs); | |
mSeekBar.setMax(mMaxValue); | |
mSeekBar.setOnSeekBarChangeListener(this); | |
setWidgetLayoutResource(R.layout.preference_seek_bar); | |
} | |
private void setValuesFromXml(AttributeSet attrs) { | |
mMaxValue = attrs.getAttributeIntValue(ANDROIDNS, "max", DEFAULT_VALUE); | |
mMinValue = attrs.getAttributeIntValue(APPLICATIONNS, "min", 0); | |
String units = getAttributeStringValue(attrs, APPLICATIONNS, "units", ""); | |
try { | |
String newInterval = attrs.getAttributeValue(APPLICATIONNS, "interval"); | |
if (newInterval != null) | |
mInterval = Integer.parseInt(newInterval); | |
} catch (Exception e) { | |
Log.e(LOG_TAG, "Invalid interval value", e); | |
} | |
} | |
private String getAttributeStringValue(AttributeSet attrs, String namespace, String name, String defaultValue) { | |
String value = attrs.getAttributeValue(namespace, name); | |
if (value == null) | |
value = defaultValue; | |
return value; | |
} | |
@Override | |
protected View onCreateView(ViewGroup parent) { | |
View view = super.onCreateView(parent); | |
// The basic preference layout puts the widget frame to the right of the title and summary, | |
// so we need to change it a bit - the seekbar should be under them. | |
LinearLayout layout = (LinearLayout) view; | |
layout.setOrientation(LinearLayout.VERTICAL); | |
return view; | |
} | |
@Override | |
public void onBindView(View view) { | |
super.onBindView(view); | |
ViewGroup newContainer = (ViewGroup) view.findViewById(R.id.seekBarPrefBarContainer); | |
ViewParent oldContainer = mSeekBar.getParent(); | |
try { | |
// move our seekbar to the new view we've been given | |
if (oldContainer != newContainer) { | |
// remove the seekbar from the old view | |
if (oldContainer != null) { | |
((ViewGroup) oldContainer).removeView(mSeekBar); | |
} | |
// remove the existing seekbar (there may not be one) and add ours | |
newContainer.removeAllViews(); | |
newContainer.addView(mSeekBar, ViewGroup.LayoutParams.MATCH_PARENT, | |
ViewGroup.LayoutParams.WRAP_CONTENT); | |
} | |
} catch (Exception ex) { | |
Log.e(LOG_TAG, "Error binding view: " + ex.toString()); | |
} | |
//if dependency is false from the beginning, disable the seek bar | |
if (view != null && !view.isEnabled()) { | |
mSeekBar.setEnabled(false); | |
} | |
updateView(view); | |
} | |
/** | |
* Update a SeekBarPreference view with our current state | |
* | |
* @param view | |
*/ | |
protected void updateView(View view) { | |
try { | |
// mStatusText = (TextView) view.findViewById(R.id.seekBarPrefValue); | |
// | |
// mStatusText.setText(String.valueOf(mCurrentValue)); | |
// mStatusText.setMinimumWidth(30); | |
mSeekBar.setProgress(mCurrentValue); | |
TextView seekMin = (TextView) view.findViewById(R.id.seekbarMinLabel); | |
seekMin.setText(String.valueOf(mMinValue)); | |
TextView seekMax = (TextView) view.findViewById(R.id.seekbarMaxLabel); | |
seekMax.setText(String.valueOf(mMaxValue)); | |
} catch (Exception e) { | |
Log.e(LOG_TAG, "Error updating seek bar preference", e); | |
} | |
} | |
@Override | |
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { | |
int newValue = progress + mMinValue; | |
if (newValue > mMaxValue) | |
newValue = mMaxValue; | |
else if (newValue < mMinValue) | |
newValue = mMinValue; | |
else if (mInterval != 1 && newValue % mInterval != 0) | |
newValue = Math.round(((float) newValue) / mInterval) * mInterval; | |
// change rejected, revert to the previous value | |
if (!callChangeListener(newValue)) { | |
seekBar.setProgress(mCurrentValue - mMinValue); | |
return; | |
} | |
// change accepted, store it | |
mCurrentValue = newValue; | |
// mStatusText.setText(String.valueOf(newValue)); | |
persistInt(newValue); | |
} | |
@Override | |
public void onStartTrackingTouch(SeekBar seekBar) { | |
} | |
@Override | |
public void onStopTrackingTouch(SeekBar seekBar) { | |
notifyChanged(); | |
} | |
@Override | |
protected Object onGetDefaultValue(TypedArray ta, int index) { | |
int defaultValue = ta.getInt(index, DEFAULT_VALUE); | |
return defaultValue; | |
} | |
@Override | |
protected void onSetInitialValue(boolean restoreValue, Object defaultValue) { | |
if (restoreValue) { | |
mCurrentValue = getPersistedInt(mCurrentValue); | |
} else { | |
int temp = 0; | |
try { | |
temp = (Integer) defaultValue; | |
} catch (Exception ex) { | |
Log.e(LOG_TAG, "Invalid default value: " + defaultValue.toString()); | |
} | |
persistInt(temp); | |
mCurrentValue = temp; | |
} | |
} | |
/** | |
* make sure that the seekbar is disabled if the preference is disabled | |
*/ | |
@Override | |
public void setEnabled(boolean enabled) { | |
super.setEnabled(enabled); | |
mSeekBar.setEnabled(enabled); | |
} | |
@Override | |
public void onDependencyChanged(Preference dependency, boolean disableDependent) { | |
super.onDependencyChanged(dependency, disableDependent); | |
//Disable movement of seek bar when dependency is false | |
if (mSeekBar != null) { | |
mSeekBar.setEnabled(!disableDependent); | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment