Compare commits
5 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| fb16805739 | |||
| a8c108aa85 | |||
| 1451fe274c | |||
| 4246b6ff8c | |||
| 26b4ce849a |
@@ -12,8 +12,8 @@ android {
|
||||
applicationId "ru.ytkab0bp.slicebeam"
|
||||
minSdk 21
|
||||
targetSdk 34
|
||||
versionCode 4
|
||||
versionName "0.1.1"
|
||||
versionCode 5
|
||||
versionName "0.1.2"
|
||||
|
||||
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
|
||||
|
||||
|
||||
@@ -66,7 +66,7 @@ public class SliceBeam extends Application {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
}
|
||||
WebView.setWebContentsDebuggingEnabled(true);
|
||||
WebView.setWebContentsDebuggingEnabled(BuildConfig.DEBUG);
|
||||
|
||||
Thread.setDefaultUncaughtExceptionHandler((t, e) -> {
|
||||
StringWriter sw = new StringWriter();
|
||||
|
||||
@@ -74,7 +74,9 @@ public abstract class UnfoldMenu {
|
||||
|
||||
private void show(View from, BedFragment fragment, FrameLayout into) {
|
||||
if (isVisible) return;
|
||||
this.fragment = fragment;
|
||||
if (fragment != null) {
|
||||
this.fragment = fragment;
|
||||
}
|
||||
this.isVisible = true;
|
||||
this.containerLayout = into;
|
||||
|
||||
@@ -96,7 +98,7 @@ public abstract class UnfoldMenu {
|
||||
fromTranslationX = pos[0] - intoPos[0];
|
||||
fromTranslationY = pos[1] - intoPos[1];
|
||||
toTranslationX = 0;
|
||||
toTranslationY = portrait ? into.getHeight() - side : 0;
|
||||
toTranslationY = portrait ? into.getHeight() - side - into.getPaddingTop() - into.getPaddingBottom() : 0;
|
||||
rootView = new FrameLayout(ctx) {
|
||||
{
|
||||
setWillNotDraw(false);
|
||||
|
||||
@@ -124,7 +124,7 @@ public class WebViewMenu extends UnfoldMenu {
|
||||
|
||||
@Override
|
||||
public int getRequestedSize(FrameLayout into, boolean portrait) {
|
||||
return portrait ? into.getHeight() : into.getWidth();
|
||||
return portrait ? into.getHeight() - into.getPaddingTop() - into.getPaddingBottom() : into.getWidth();
|
||||
}
|
||||
|
||||
private final class Bridge {
|
||||
|
||||
@@ -18,6 +18,8 @@ import androidx.core.graphics.ColorUtils;
|
||||
import androidx.recyclerview.widget.RecyclerView;
|
||||
|
||||
import java.io.ByteArrayOutputStream;
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.InputStream;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.util.ArrayList;
|
||||
@@ -36,13 +38,16 @@ import ru.ytkab0bp.slicebeam.components.UnfoldMenu;
|
||||
import ru.ytkab0bp.slicebeam.components.WebViewMenu;
|
||||
import ru.ytkab0bp.slicebeam.config.ConfigObject;
|
||||
import ru.ytkab0bp.slicebeam.events.NeedDismissCalibrationsMenu;
|
||||
import ru.ytkab0bp.slicebeam.events.NeedSnackbarEvent;
|
||||
import ru.ytkab0bp.slicebeam.events.ObjectsListChangedEvent;
|
||||
import ru.ytkab0bp.slicebeam.events.SelectedObjectChangedEvent;
|
||||
import ru.ytkab0bp.slicebeam.fragment.BedFragment;
|
||||
import ru.ytkab0bp.slicebeam.recycler.PreferenceItem;
|
||||
import ru.ytkab0bp.slicebeam.recycler.SimpleRecyclerAdapter;
|
||||
import ru.ytkab0bp.slicebeam.recycler.SimpleRecyclerItem;
|
||||
import ru.ytkab0bp.slicebeam.recycler.SpaceItem;
|
||||
import ru.ytkab0bp.slicebeam.slic3r.Bed3D;
|
||||
import ru.ytkab0bp.slicebeam.slic3r.Slic3rRuntimeError;
|
||||
import ru.ytkab0bp.slicebeam.theme.BeamTheme;
|
||||
import ru.ytkab0bp.slicebeam.theme.ThemesRepo;
|
||||
import ru.ytkab0bp.slicebeam.utils.ViewUtils;
|
||||
@@ -214,7 +219,7 @@ public class FileMenu extends ListBedMenu {
|
||||
public final class CalibrationsMenu extends UnfoldMenu {
|
||||
|
||||
public int getRequestedSize(FrameLayout into, boolean portrait) {
|
||||
return (int) (portrait ? into.getHeight() * 0.3f : into.getWidth() * 0.6f);
|
||||
return (int) (portrait ? into.getHeight() * 0.35f : into.getWidth() * 0.6f);
|
||||
}
|
||||
|
||||
private String loadJSLoader(String key) {
|
||||
@@ -309,6 +314,11 @@ public class FileMenu extends ListBedMenu {
|
||||
if (ctx instanceof MainActivity) {
|
||||
((MainActivity) ctx).showUnfoldMenu(new WebViewMenu(Uri.parse("https://k3d.tech/calibrations/retractions/calibrator/").buildUpon().appendQueryParameter("lang", getK3DLanguage()).build(), loadJSLoader("k3d_rct")).setFragment(fragment), v);
|
||||
}
|
||||
}),
|
||||
new PreferenceItem().setIcon(R.drawable.deployed_code_24).setTitle(ctx.getString(R.string.MenuFileCalibrationsModels)).setSubtitle(ctx.getString(R.string.MenuFileCalibrationsModelsDescription)).setOnClickListener(v -> {
|
||||
if (ctx instanceof MainActivity) {
|
||||
((MainActivity) ctx).showUnfoldMenu(new CalibrationModelsMenu().setFragment(fragment), v);
|
||||
}
|
||||
})
|
||||
));
|
||||
rv.setAdapter(adapter);
|
||||
@@ -359,4 +369,111 @@ public class FileMenu extends ListBedMenu {
|
||||
SliceBeam.EVENT_BUS.unregisterListener(this);
|
||||
}
|
||||
}
|
||||
|
||||
public final static class CalibrationModelsMenu extends UnfoldMenu {
|
||||
private void loadModel(String key) {
|
||||
BedFragment fragment = this.fragment;
|
||||
ViewUtils.postOnMainThread(() -> {
|
||||
File f = new File(SliceBeam.getModelCacheDir(), "calibration_" + key + ".stl");
|
||||
new Thread(()->{
|
||||
try {
|
||||
InputStream in = SliceBeam.INSTANCE.getAssets().open("models/" + key + ".stl");
|
||||
FileOutputStream fos = new FileOutputStream(f);
|
||||
byte[] buffer = new byte[10240]; int c;
|
||||
while ((c = in.read(buffer)) != -1) {
|
||||
fos.write(buffer, 0, c);
|
||||
}
|
||||
fos.close();
|
||||
in.close();
|
||||
|
||||
ViewUtils.postOnMainThread(() -> {
|
||||
try {
|
||||
if (f.getName().endsWith(".gcode")) {
|
||||
fragment.loadGCode(f);
|
||||
} else {
|
||||
fragment.loadModel(f);
|
||||
SliceBeam.EVENT_BUS.fireEvent(new ObjectsListChangedEvent());
|
||||
}
|
||||
SliceBeam.EVENT_BUS.fireEvent(new NeedSnackbarEvent(R.string.MenuFileOpenFileLoaded));
|
||||
} catch (Slic3rRuntimeError e) {
|
||||
f.delete();
|
||||
|
||||
ViewUtils.postOnMainThread(() -> new BeamAlertDialogBuilder(fragment.getContext())
|
||||
.setTitle(R.string.MenuFileOpenFileFailed)
|
||||
.setMessage(e.toString())
|
||||
.setPositiveButton(android.R.string.ok, null)
|
||||
.show());
|
||||
}
|
||||
});
|
||||
} catch (Exception e) {
|
||||
f.delete();
|
||||
ViewUtils.postOnMainThread(() -> new BeamAlertDialogBuilder(fragment.getContext())
|
||||
.setTitle(R.string.MenuFileOpenFileFailed)
|
||||
.setMessage(e.toString())
|
||||
.setPositiveButton(android.R.string.ok, null)
|
||||
.show());
|
||||
}
|
||||
}).start();
|
||||
}, 200);
|
||||
SliceBeam.EVENT_BUS.fireEvent(new NeedDismissCalibrationsMenu());
|
||||
dismiss(true);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected View onCreateView(Context ctx, boolean portrait) {
|
||||
LinearLayout ll = new LinearLayout(ctx);
|
||||
ll.setOrientation(LinearLayout.VERTICAL);
|
||||
|
||||
LinearLayout toolbar = new LinearLayout(ctx);
|
||||
toolbar.setPadding(ViewUtils.dp(12), 0, ViewUtils.dp(12), 0);
|
||||
toolbar.setOrientation(LinearLayout.HORIZONTAL);
|
||||
toolbar.setGravity(Gravity.CENTER_VERTICAL);
|
||||
toolbar.setBackground(ViewUtils.createRipple(ThemesRepo.getColor(android.R.attr.colorControlHighlight), 0));
|
||||
toolbar.setOnClickListener(v -> dismiss());
|
||||
|
||||
ImageView icon = new ImageView(ctx);
|
||||
icon.setImageResource(R.drawable.arrow_left_outline_28);
|
||||
icon.setColorFilter(ThemesRepo.getColor(android.R.attr.textColorSecondary));
|
||||
toolbar.addView(icon, new LinearLayout.LayoutParams(ViewUtils.dp(28), ViewUtils.dp(28)));
|
||||
|
||||
TextView title = new TextView(ctx);
|
||||
title.setText(R.string.MenuOrientationPositionBack);
|
||||
title.setTypeface(ViewUtils.getTypeface(ViewUtils.ROBOTO_MEDIUM));
|
||||
title.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 18);
|
||||
title.setTextColor(ThemesRepo.getColor(android.R.attr.textColorPrimary));
|
||||
toolbar.addView(title, new LinearLayout.LayoutParams(0, ViewGroup.LayoutParams.WRAP_CONTENT, 1f) {{
|
||||
leftMargin = ViewUtils.dp(12);
|
||||
}});
|
||||
ll.addView(toolbar, new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewUtils.dp(52)));
|
||||
|
||||
ll.addView(new DividerView(ctx), new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewUtils.dp(1f)));
|
||||
|
||||
RecyclerView rv = new FadeRecyclerView(ctx);
|
||||
SimpleRecyclerAdapter adapter = new SimpleRecyclerAdapter();
|
||||
adapter.setItems(Arrays.asList(
|
||||
new PreferenceItem().setIcon(R.drawable.model_3dbenchy).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModels3DBenchy)).setOnClickListener(v -> loadModel("3dbenchy")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_xyz_cube).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsXYZCube)).setOnClickListener(v -> loadModel("xyz_cube")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_bunny).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsBunny)).setOnClickListener(v -> loadModel("bunny")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_fox).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsFox)).setOnClickListener(v -> loadModel("fox")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_box).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsBox)).setOnClickListener(v -> loadModel("box")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_cone).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsCone)).setOnClickListener(v -> loadModel("cone")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_cylinder).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsCylinder)).setOnClickListener(v -> loadModel("cylinder")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_pyramid).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsPyramid)).setOnClickListener(v -> loadModel("pyramid")),
|
||||
new PreferenceItem().setIcon(R.drawable.model_sphere).setNoTint(true).setRoundRadius(ViewUtils.dp(8)).setTitle(ctx.getString(R.string.MenuFileCalibrationsModelsSphere)).setOnClickListener(v -> loadModel("sphere"))
|
||||
));
|
||||
rv.setAdapter(adapter);
|
||||
ll.addView(rv, new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, 0, 1f));
|
||||
return ll;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getRequestedSize(FrameLayout into, boolean portrait) {
|
||||
return portrait ? into.getHeight() - into.getPaddingTop() - into.getPaddingBottom() : into.getWidth();
|
||||
}
|
||||
|
||||
public CalibrationModelsMenu setFragment(BedFragment fragment) {
|
||||
this.fragment = fragment;
|
||||
return this;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -2,6 +2,7 @@ package ru.ytkab0bp.slicebeam.components.bed_menu;
|
||||
|
||||
import static ru.ytkab0bp.slicebeam.utils.DebugUtils.assertTrue;
|
||||
|
||||
import android.annotation.SuppressLint;
|
||||
import android.app.Activity;
|
||||
import android.content.Context;
|
||||
import android.content.Intent;
|
||||
@@ -35,6 +36,7 @@ import org.json.JSONObject;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.FileNotFoundException;
|
||||
import java.text.DecimalFormat;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Arrays;
|
||||
import java.util.Collections;
|
||||
@@ -53,6 +55,7 @@ import ru.ytkab0bp.slicebeam.config.ConfigObject;
|
||||
import ru.ytkab0bp.slicebeam.events.NeedSnackbarEvent;
|
||||
import ru.ytkab0bp.slicebeam.fragment.BedFragment;
|
||||
import ru.ytkab0bp.slicebeam.recycler.SimpleRecyclerItem;
|
||||
import ru.ytkab0bp.slicebeam.slic3r.GCodeProcessorResult;
|
||||
import ru.ytkab0bp.slicebeam.slic3r.GCodeViewer;
|
||||
import ru.ytkab0bp.slicebeam.slic3r.Slic3rLocalization;
|
||||
import ru.ytkab0bp.slicebeam.theme.IThemeView;
|
||||
@@ -187,11 +190,16 @@ public class SliceMenu extends ListBedMenu {
|
||||
private TextView totalView;
|
||||
private SegmentsView segmentsView;
|
||||
private ExtrusionRoleView[] roleViews = new ExtrusionRoleView[GCodeViewer.EXTRUSION_ROLES_COUNT];
|
||||
private static DecimalFormat format = new DecimalFormat("0.##");
|
||||
|
||||
private GCodeViewer getViewer() {
|
||||
return fragment.getGlView().getRenderer().getViewer();
|
||||
}
|
||||
|
||||
private GCodeProcessorResult getResult() {
|
||||
return fragment.getGlView().getRenderer().getGcodeResult();
|
||||
}
|
||||
|
||||
@Override
|
||||
protected View onCreateView(Context ctx, boolean portrait) {
|
||||
LinearLayout ll = new LinearLayout(ctx);
|
||||
@@ -250,12 +258,13 @@ public class SliceMenu extends ListBedMenu {
|
||||
super.onCreate();
|
||||
|
||||
GCodeViewer viewer = getViewer();
|
||||
GCodeProcessorResult result = getResult();
|
||||
if (viewer != null) {
|
||||
for (int i = 0; i < GCodeViewer.EXTRUSION_ROLES_COUNT; i++) {
|
||||
boolean visible = viewer.getEstimatedTime(i) != 0;
|
||||
roleViews[i].setVisibility(visible ? View.VISIBLE : View.GONE);
|
||||
if (visible) {
|
||||
roleViews[i].bind(viewer, i);
|
||||
roleViews[i].bind(viewer, result, i);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -299,22 +308,28 @@ public class SliceMenu extends ListBedMenu {
|
||||
return total;
|
||||
}
|
||||
|
||||
@SuppressLint("SetTextI18n")
|
||||
private void updateValues() {
|
||||
GCodeViewer viewer = getViewer();
|
||||
GCodeProcessorResult result = getResult();
|
||||
if (viewer == null) return;
|
||||
|
||||
float[] values = new float[2 + GCodeViewer.EXTRUSION_ROLES_COUNT];
|
||||
double totalWeight = 0;
|
||||
double totalLength = 0;
|
||||
values[0] = 0;
|
||||
values[values.length - 1] = 1;
|
||||
float prev = 0;
|
||||
int lastVisible = 0;
|
||||
float total = getTotalEstimatedTime();
|
||||
float totalTime = getTotalEstimatedTime();
|
||||
for (int i = 0; i < GCodeViewer.EXTRUSION_ROLES_COUNT; i++) {
|
||||
if (viewer.isExtrusionRoleVisible(i)) {
|
||||
float percent = viewer.getEstimatedTime(i) / total;
|
||||
float percent = viewer.getEstimatedTime(i) / totalTime;
|
||||
values[i + 1] = prev + percent;
|
||||
lastVisible = i;
|
||||
prev = values[i + 1];
|
||||
totalLength += result.getUsedFilamentMM(i);
|
||||
totalWeight += result.getUsedFilamentG(i);
|
||||
} else {
|
||||
values[i + 1] = prev;
|
||||
}
|
||||
@@ -323,11 +338,21 @@ public class SliceMenu extends ListBedMenu {
|
||||
values[lastVisible] = 1;
|
||||
|
||||
segmentsView.setValues(values);
|
||||
totalView.setText(formatTime(total));
|
||||
totalView.setText(formatComplex(totalWeight, totalLength, totalTime));
|
||||
}
|
||||
|
||||
private static String formatComplex(double weight, double length, float time) {
|
||||
StringBuilder sb = new StringBuilder();
|
||||
if (weight > 0) {
|
||||
sb.append(format.format(weight)).append(" ").append(SliceBeam.INSTANCE.getString(R.string.MenuSliceInfoWeight)).append(" | ");
|
||||
}
|
||||
sb.append(format.format(length)).append(" ").append(SliceBeam.INSTANCE.getString(R.string.MenuSliceInfoLength)).append(" | ");
|
||||
sb.append(formatTime(time));
|
||||
return sb.toString();
|
||||
}
|
||||
|
||||
private static String formatTime(float time) {
|
||||
int secondsTotal = Math.round(time);
|
||||
int secondsTotal = (int) Math.round(Math.ceil(time));
|
||||
int seconds = secondsTotal % 60;
|
||||
int minutes = ((secondsTotal - seconds) / 60) % 60;
|
||||
int hours = ((secondsTotal - seconds) / 60) / 60;
|
||||
@@ -341,7 +366,7 @@ public class SliceMenu extends ListBedMenu {
|
||||
|
||||
sb.append(minutes).append(" ").append(SliceBeam.INSTANCE.getString(R.string.MenuSliceInfoMinute));
|
||||
}
|
||||
if (seconds > 0) {
|
||||
if (seconds > 0 || sb.length() == 0) {
|
||||
if (sb.length() > 0) sb.append(" ");
|
||||
|
||||
sb.append(seconds).append(" ").append(SliceBeam.INSTANCE.getString(R.string.MenuSliceInfoSecond));
|
||||
@@ -379,18 +404,19 @@ public class SliceMenu extends ListBedMenu {
|
||||
titleView = new TextView(context);
|
||||
titleView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 15);
|
||||
addView(titleView, new LinearLayout.LayoutParams(0, ViewGroup.LayoutParams.WRAP_CONTENT, 1f) {{
|
||||
leftMargin = rightMargin = ViewUtils.dp(12);
|
||||
leftMargin = ViewUtils.dp(12);
|
||||
rightMargin = ViewUtils.dp(8);
|
||||
}});
|
||||
|
||||
timeView = new TextView(context);
|
||||
timeView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 13);
|
||||
timeView.setTextSize(TypedValue.COMPLEX_UNIT_DIP, 12);
|
||||
addView(timeView, new LinearLayout.LayoutParams(ViewGroup.LayoutParams.WRAP_CONTENT, ViewGroup.LayoutParams.WRAP_CONTENT));
|
||||
|
||||
setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewUtils.dp(42)));
|
||||
onApplyTheme();
|
||||
}
|
||||
|
||||
public void bind(GCodeViewer viewer, @GCodeViewer.ExtrusionRole int role) {
|
||||
public void bind(GCodeViewer viewer, GCodeProcessorResult result, @GCodeViewer.ExtrusionRole int role) {
|
||||
switch (role) {
|
||||
case GCodeViewer.EXTRUSION_ROLE_NONE:
|
||||
titleView.setText(Slic3rLocalization.getString("Unknown"));
|
||||
@@ -439,7 +465,7 @@ public class SliceMenu extends ListBedMenu {
|
||||
break;
|
||||
}
|
||||
|
||||
timeView.setText(formatTime(viewer.getEstimatedTime(role)));
|
||||
timeView.setText(formatComplex(result.getUsedFilamentG(role), result.getUsedFilamentMM(role), viewer.getEstimatedTime(role)));
|
||||
|
||||
checkBox.setChecked(viewer.isExtrusionRoleVisible(role));
|
||||
checkBox.setButtonTintList(ColorStateList.valueOf(SegmentsView.mapColor(role)));
|
||||
|
||||
@@ -39,13 +39,15 @@ public class FilamentConfigFragment extends ProfileListFragment {
|
||||
|
||||
List<ConfigObject> nList = new ArrayList<>(list.size());
|
||||
Slic3rUtils.ConfigChecker checker = new Slic3rUtils.ConfigChecker(SliceBeam.CONFIG.findPrinter(printer).serialize());
|
||||
Slic3rUtils.ConfigChecker printChecker = new Slic3rUtils.ConfigChecker(SliceBeam.CONFIG.findPrint(print).serialize());
|
||||
for (ConfigObject obj : list) {
|
||||
if (checker.checkCompatibility(obj.get("compatible_printers_condition")) && printChecker.checkCompatibility(obj.get("compatible_prints_condition"))) {
|
||||
nList.add(obj);
|
||||
if (SliceBeam.CONFIG.findPrint(print) != null) {
|
||||
Slic3rUtils.ConfigChecker printChecker = new Slic3rUtils.ConfigChecker(SliceBeam.CONFIG.findPrint(print).serialize());
|
||||
for (ConfigObject obj : list) {
|
||||
if (checker.checkCompatibility(obj.get("compatible_printers_condition")) && printChecker.checkCompatibility(obj.get("compatible_prints_condition"))) {
|
||||
nList.add(obj);
|
||||
}
|
||||
}
|
||||
printChecker.release();
|
||||
}
|
||||
printChecker.release();
|
||||
checker.release();
|
||||
lastPrinter = printer;
|
||||
lastPrint = print;
|
||||
|
||||
@@ -2,6 +2,8 @@ package ru.ytkab0bp.slicebeam.recycler;
|
||||
|
||||
import android.content.Context;
|
||||
import android.content.res.ColorStateList;
|
||||
import android.graphics.Canvas;
|
||||
import android.graphics.Path;
|
||||
import android.graphics.Typeface;
|
||||
import android.graphics.drawable.Drawable;
|
||||
import android.text.TextUtils;
|
||||
@@ -13,6 +15,8 @@ import android.widget.ImageView;
|
||||
import android.widget.LinearLayout;
|
||||
import android.widget.TextView;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
import androidx.appcompat.widget.AppCompatImageView;
|
||||
import androidx.core.content.ContextCompat;
|
||||
import androidx.recyclerview.widget.RecyclerView;
|
||||
|
||||
@@ -29,6 +33,7 @@ public class PreferenceItem extends SimpleRecyclerItem<PreferenceItem.Preference
|
||||
private int textColorRes;
|
||||
private boolean noTint;
|
||||
private ValueProvider valueProvider;
|
||||
private float roundRadius;
|
||||
|
||||
public PreferenceItem setTitle(CharSequence title) {
|
||||
mTitle = title;
|
||||
@@ -70,6 +75,11 @@ public class PreferenceItem extends SimpleRecyclerItem<PreferenceItem.Preference
|
||||
return this;
|
||||
}
|
||||
|
||||
public PreferenceItem setRoundRadius(float roundRadius) {
|
||||
this.roundRadius = roundRadius;
|
||||
return this;
|
||||
}
|
||||
|
||||
public PreferenceItem setTextColorRes(int textColorRes) {
|
||||
this.textColorRes = textColorRes;
|
||||
return this;
|
||||
@@ -94,6 +104,7 @@ public class PreferenceItem extends SimpleRecyclerItem<PreferenceItem.Preference
|
||||
private TextView title, subtitle;
|
||||
private ImageView icon;
|
||||
private TextView value;
|
||||
private float radius;
|
||||
|
||||
public PreferenceHolderView(Context context) {
|
||||
super(context);
|
||||
@@ -101,7 +112,23 @@ public class PreferenceItem extends SimpleRecyclerItem<PreferenceItem.Preference
|
||||
setOrientation(HORIZONTAL);
|
||||
setGravity(Gravity.CENTER_VERTICAL);
|
||||
|
||||
icon = new ImageView(context);
|
||||
icon = new AppCompatImageView(context) {
|
||||
private Path path = new Path();
|
||||
|
||||
@Override
|
||||
public void draw(@NonNull Canvas canvas) {
|
||||
if (radius != 0) {
|
||||
canvas.save();
|
||||
path.rewind();
|
||||
path.addRoundRect(0, 0, getWidth(), getHeight(), radius, radius, Path.Direction.CW);
|
||||
canvas.clipPath(path);
|
||||
}
|
||||
super.draw(canvas);
|
||||
if (radius != 0) {
|
||||
canvas.restore();
|
||||
}
|
||||
}
|
||||
};
|
||||
icon.setLayoutParams(new LayoutParams(ViewUtils.dp(28), ViewUtils.dp(28)) {{
|
||||
setMarginStart(ViewUtils.dp(4));
|
||||
setMarginEnd(ViewUtils.dp(8));
|
||||
@@ -178,6 +205,11 @@ public class PreferenceItem extends SimpleRecyclerItem<PreferenceItem.Preference
|
||||
} else {
|
||||
icon.setImageTintList(ColorStateList.valueOf(ThemesRepo.getColor(item.textColorRes != 0 ? item.textColorRes : android.R.attr.textColorSecondary)));
|
||||
}
|
||||
radius = item.roundRadius;
|
||||
icon.invalidate();
|
||||
|
||||
ViewGroup.LayoutParams params = icon.getLayoutParams();
|
||||
params.width = params.height = radius != 0 ? ViewUtils.dp(42) : ViewUtils.dp(28);
|
||||
}
|
||||
|
||||
@Override
|
||||
|
||||
@@ -13,6 +13,14 @@ public class GCodeProcessorResult {
|
||||
pointer = ptr;
|
||||
}
|
||||
|
||||
public double getUsedFilamentMM(@GCodeViewer.ExtrusionRole int role) {
|
||||
return Native.gcoderesult_get_used_filament_mm(pointer, role);
|
||||
}
|
||||
|
||||
public double getUsedFilamentG(@GCodeViewer.ExtrusionRole int role) {
|
||||
return Native.gcoderesult_get_used_filament_g(pointer, role);
|
||||
}
|
||||
|
||||
public String getRecommendedName() {
|
||||
return Native.gcoderesult_get_recommended_name(pointer);
|
||||
}
|
||||
|
||||
@@ -73,6 +73,8 @@ class Native {
|
||||
|
||||
static native long gcoderesult_load_file(String path, String name);
|
||||
static native String gcoderesult_get_recommended_name(long ptr);
|
||||
static native double gcoderesult_get_used_filament_mm(long ptr, int role);
|
||||
static native double gcoderesult_get_used_filament_g(long ptr, int role);
|
||||
static native void gcoderesult_release(long ptr);
|
||||
|
||||
static native long vgcode_create();
|
||||
|
||||
@@ -637,6 +637,73 @@ extern "C" {
|
||||
return env->NewStringUTF(ref->name.c_str());
|
||||
}
|
||||
|
||||
GCodeExtrusionRole mapGCodeRole(int index) {
|
||||
GCodeExtrusionRole gRole;
|
||||
switch (index) {
|
||||
default:
|
||||
case 0:
|
||||
gRole = GCodeExtrusionRole::None;
|
||||
break;
|
||||
case 1:
|
||||
gRole = GCodeExtrusionRole::Perimeter;
|
||||
break;
|
||||
case 2:
|
||||
gRole = GCodeExtrusionRole::ExternalPerimeter;
|
||||
break;
|
||||
case 3:
|
||||
gRole = GCodeExtrusionRole::OverhangPerimeter;
|
||||
break;
|
||||
case 4:
|
||||
gRole = GCodeExtrusionRole::InternalInfill;
|
||||
break;
|
||||
case 5:
|
||||
gRole = GCodeExtrusionRole::SolidInfill;
|
||||
break;
|
||||
case 6:
|
||||
gRole = GCodeExtrusionRole::TopSolidInfill;
|
||||
break;
|
||||
case 7:
|
||||
gRole = GCodeExtrusionRole::Ironing;
|
||||
break;
|
||||
case 8:
|
||||
gRole = GCodeExtrusionRole::BridgeInfill;
|
||||
break;
|
||||
case 9:
|
||||
gRole = GCodeExtrusionRole::GapFill;
|
||||
break;
|
||||
case 10:
|
||||
gRole = GCodeExtrusionRole::Skirt;
|
||||
break;
|
||||
case 11:
|
||||
gRole = GCodeExtrusionRole::SupportMaterial;
|
||||
break;
|
||||
case 12:
|
||||
gRole = GCodeExtrusionRole::SupportMaterialInterface;
|
||||
break;
|
||||
case 13:
|
||||
gRole = GCodeExtrusionRole::WipeTower;
|
||||
break;
|
||||
case 14:
|
||||
gRole = GCodeExtrusionRole::Custom;
|
||||
break;
|
||||
}
|
||||
return gRole;
|
||||
}
|
||||
|
||||
JNIEXPORT jdouble JNICALL Java_ru_ytkab0bp_slicebeam_slic3r_Native_gcoderesult_1get_1used_1filament_1mm(JNIEnv* env, jclass, jlong ptr, jint role) {
|
||||
GCodeResultRef* ref = (GCodeResultRef*) (intptr_t) ptr;
|
||||
|
||||
std::pair<double, double> info = ref->result.print_statistics.used_filaments_per_role.find(mapGCodeRole(role))->second;
|
||||
return info.first * 1000.0 / 25.4;
|
||||
}
|
||||
|
||||
JNIEXPORT jdouble JNICALL Java_ru_ytkab0bp_slicebeam_slic3r_Native_gcoderesult_1get_1used_1filament_1g(JNIEnv* env, jclass, jlong ptr, jint role) {
|
||||
GCodeResultRef* ref = (GCodeResultRef*) (intptr_t) ptr;
|
||||
|
||||
std::pair<double, double> info = ref->result.print_statistics.used_filaments_per_role.find(mapGCodeRole(role))->second;
|
||||
return info.second;
|
||||
}
|
||||
|
||||
JNIEXPORT void JNICALL Java_ru_ytkab0bp_slicebeam_slic3r_Native_gcoderesult_1release(JNIEnv* env, jclass, jlong ptr) {
|
||||
GCodeResultRef* ref = (GCodeResultRef*) (intptr_t) ptr;
|
||||
delete ref;
|
||||
|
||||
@@ -0,0 +1,10 @@
|
||||
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||
android:width="24dp"
|
||||
android:height="24dp"
|
||||
android:viewportWidth="960"
|
||||
android:viewportHeight="960"
|
||||
android:tint="#000">
|
||||
<path
|
||||
android:fillColor="@android:color/white"
|
||||
android:pathData="M440,777L440,503L200,364L200,638Q200,638 200,638Q200,638 200,638L440,777ZM520,777L760,638Q760,638 760,638Q760,638 760,638L760,364L520,503L520,777ZM480,434L717,297L480,160Q480,160 480,160Q480,160 480,160L243,297L480,434ZM160,708Q141,697 130.5,679Q120,661 120,639L120,321Q120,299 130.5,281Q141,263 160,252L440,91Q459,80 480,80Q501,80 520,91L800,252Q819,263 829.5,281Q840,299 840,321L840,639Q840,661 829.5,679Q819,697 800,708L520,869Q501,880 480,880Q459,880 440,869L160,708ZM480,480Q480,480 480,480Q480,480 480,480L480,480Q480,480 480,480Q480,480 480,480L480,480Q480,480 480,480Q480,480 480,480L480,480Q480,480 480,480Q480,480 480,480L480,480Q480,480 480,480Q480,480 480,480L480,480Q480,480 480,480Q480,480 480,480Z"/>
|
||||
</vector>
|
||||
|
After Width: | Height: | Size: 210 KiB |
|
After Width: | Height: | Size: 79 KiB |
|
After Width: | Height: | Size: 153 KiB |
|
After Width: | Height: | Size: 100 KiB |
|
After Width: | Height: | Size: 78 KiB |
|
After Width: | Height: | Size: 135 KiB |
|
After Width: | Height: | Size: 77 KiB |
|
After Width: | Height: | Size: 137 KiB |
|
After Width: | Height: | Size: 86 KiB |
@@ -18,6 +18,16 @@
|
||||
<string name="MenuFileCalibrationsLADescription">Калибровка Linear/Pressure Advance</string>
|
||||
<string name="MenuFileCalibrationsRetract">K3D Откаты</string>
|
||||
<string name="MenuFileCalibrationsRetractDescription">Калибровка расстояния и скорости откатов</string>
|
||||
<string name="MenuFileCalibrationsModels">Калибровочные модели</string>
|
||||
<string name="MenuFileCalibrationsModelsDescription">Библиотека калибровочных моделей</string>
|
||||
<string name="MenuFileCalibrationsModelsXYZCube">Калибровочный куб</string>
|
||||
<string name="MenuFileCalibrationsModelsBunny">Заяц</string>
|
||||
<string name="MenuFileCalibrationsModelsFox">Лис</string>
|
||||
<string name="MenuFileCalibrationsModelsBox">Куб</string>
|
||||
<string name="MenuFileCalibrationsModelsCone">Конус</string>
|
||||
<string name="MenuFileCalibrationsModelsCylinder">Цилиндр</string>
|
||||
<string name="MenuFileCalibrationsModelsPyramid">Пирамида</string>
|
||||
<string name="MenuFileCalibrationsModelsSphere">Сфера</string>
|
||||
<string name="MenuFileOpenFileLoaded">Файл загружен.</string>
|
||||
<string name="MenuFileImportProfiles">Импорт. профилей</string>
|
||||
<string name="MenuFileImportProfilesFailed">Не удалось импортировать</string>
|
||||
@@ -72,6 +82,8 @@
|
||||
<string name="MenuSliceInfoSecond">с.</string>
|
||||
<string name="MenuSliceInfoMinute">мин.</string>
|
||||
<string name="MenuSliceInfoHour">ч.</string>
|
||||
<string name="MenuSliceInfoWeight">г.</string>
|
||||
<string name="MenuSliceInfoLength">мм.</string>
|
||||
<string name="MenuSliceLayers">Слои</string>
|
||||
<string name="MenuSliceInfoLayers">Слои %d - %d</string>
|
||||
<string name="MenuSliceExportToFile">Экспорт. в файл</string>
|
||||
|
||||
@@ -20,6 +20,17 @@
|
||||
<string name="MenuFileCalibrationsLADescription">Linear/Pressure Advance Calibration</string>
|
||||
<string name="MenuFileCalibrationsRetract">K3D Retractions</string>
|
||||
<string name="MenuFileCalibrationsRetractDescription">Retraction distance and speed calibration</string>
|
||||
<string name="MenuFileCalibrationsModels">Calibration models</string>
|
||||
<string name="MenuFileCalibrationsModelsDescription">Calibration models library</string>
|
||||
<string name="MenuFileCalibrationsModels3DBenchy" translatable="false">3D Benchy</string>
|
||||
<string name="MenuFileCalibrationsModelsXYZCube">Calibration cube</string>
|
||||
<string name="MenuFileCalibrationsModelsBunny">Bunny</string>
|
||||
<string name="MenuFileCalibrationsModelsFox">Fox</string>
|
||||
<string name="MenuFileCalibrationsModelsBox">Box</string>
|
||||
<string name="MenuFileCalibrationsModelsCone">Cone</string>
|
||||
<string name="MenuFileCalibrationsModelsCylinder">Cylinder</string>
|
||||
<string name="MenuFileCalibrationsModelsPyramid">Pyramid</string>
|
||||
<string name="MenuFileCalibrationsModelsSphere">Sphere</string>
|
||||
<string name="MenuFileImportProfiles">Import profiles</string>
|
||||
<string name="MenuFileImportProfilesFailed">Import failed</string>
|
||||
<string name="MenuFileImportProfilesFailedNotIni">Not an .ini file</string>
|
||||
@@ -73,6 +84,8 @@
|
||||
<string name="MenuSliceInfoSecond">s.</string>
|
||||
<string name="MenuSliceInfoMinute">min.</string>
|
||||
<string name="MenuSliceInfoHour">h.</string>
|
||||
<string name="MenuSliceInfoWeight">g.</string>
|
||||
<string name="MenuSliceInfoLength">mm.</string>
|
||||
<string name="MenuSliceLayers">Layers</string>
|
||||
<string name="MenuSliceInfoLayers">Layers %d - %d</string>
|
||||
<string name="MenuSliceExportToFile">Export to file</string>
|
||||
|
||||