当前位置: 首页 > news >正文

android设置添加设备QR码信息

摘要:客户衍生需求,通过扫QR码快速获取设备基础信息,并且基于POS SDK进行打印。

1. 定位至device info的xml添加相关perference

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml	(版本 347)
@@ -92,10 +92,17 @@android:selectable="false"android:title="@string/my_device_info_device_details_category_title">+        <com.android.settings.widget.QRDeviceInfoPreference
+            android:key="qr_info"
+            android:order="18"
+            android:persistent="false"
+            android:selectable="false"
+            settings:controller="com.android.settings.deviceinfo.QRDeviceInfoPreferenceController" />
+<!-- SIM status --><Preferenceandroid:key="sim_status"
-            android:order="18"
+            android:order="19"android:title="@string/sim_status_title"settings:keywords="@string/keywords_sim_status"android:summary="@string/summary_placeholder"
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/aboutphone/MyDeviceInfoFragment.java
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/aboutphone/MyDeviceInfoFragment.java	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/aboutphone/MyDeviceInfoFragment.java	(版本 347)
@@ -41,6 +41,7 @@
+import com.android.settings.deviceinfo.QRDeviceInfoPreferenceController;
@@ -150,6 +151,7 @@
+        controllers.add(new QRDeviceInfoPreferenceController(context));return controllers;}

2. preference布局

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/layout/qr_device_info_preference.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/layout/qr_device_info_preference.xml	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/layout/qr_device_info_preference.xml	(版本 347)
@@ -0,0 +1,23 @@
+<?xml version="1.0" encoding="utf-8"?>
+<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
+    android:layout_width="match_parent"
+    android:layout_height="wrap_content"
+    android:orientation="vertical" >
+
+    <ImageView
+        android:id="@+id/iv_qr_image"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:layout_gravity="center"/>
+
+    <Button
+        android:id="@+id/print_button"
+        android:layout_width="wrap_content"
+        android:layout_height="wrap_content"
+        android:layout_gravity="bottom|end"
+        android:paddingVertical="14dp"
+        android:drawableStart="@drawable/ic_print"
+        android:drawablePadding="9dp"
+        android:text="@string/print"
+        style="@style/ActionPrimaryButton"/>
+</FrameLayout>
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/drawable/ic_print.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/drawable/ic_print.xml	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/drawable/ic_print.xml	(版本 347)
@@ -0,0 +1,26 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+  ~ Copyright 2017 The Android Open Source Project
+  ~
+  ~ Licensed under the Apache License, Version 2.0 (the "License");
+  ~ you may not use this file except in compliance with the License.
+  ~ You may obtain a copy of the License at
+  ~
+  ~      http://www.apache.org/licenses/LICENSE-2.0
+  ~
+  ~ Unless required by applicable law or agreed to in writing, software
+  ~ distributed under the License is distributed on an "AS IS" BASIS,
+  ~ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+  ~ See the License for the specific language governing permissions and
+  ~ limitations under the License.
+  -->
+
+<vector xmlns:android="http://schemas.android.com/apk/res/android"
+    android:width="24dp"
+    android:height="24dp"
+    android:viewportWidth="24.0"
+    android:viewportHeight="24.0">
+    <path
+        android:pathData="M19,8L5,8c-1.66,0 -3,1.34 -3,3v6h4v4h12v-4h4v-6c0,-1.66 -1.34,-3 -3,-3zM16,19L8,19v-5h8v5zM19,12c-0.55,0 -1,-0.45 -1,-1s0.45,-1 1,-1 1,0.45 1,1 -0.45,1 -1,1zM18,3L6,3v4h12L18,3z"
+        android:fillColor="#FFFFFF"/>
+</vector>

3. 实现

QR码的生成使用的是google公开的com.google.zxing

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreference.java
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreference.java	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreference.java	(版本 347)
@@ -0,0 +1,401 @@
+/*
+ * Copyright (C) 2017 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *      http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package com.android.settings.widget;
+
+import android.app.AlertDialog;
+import android.content.Context;
+import android.content.DialogInterface;
+import android.graphics.Bitmap;
+import android.graphics.BitmapFactory;
+import android.graphics.Canvas;
+import android.os.Build;
+import android.os.Environment;
+import android.os.Handler;
+import android.os.Message;
+import android.os.SystemProperties;
+import android.telephony.TelephonyManager;
+import android.text.TextUtils;
+import android.util.AttributeSet;
+import android.util.Log;
+import android.view.LayoutInflater;
+import android.view.View;
+import android.widget.Button;
+import android.widget.ImageView;
+
+import androidx.preference.Preference;
+import androidx.preference.PreferenceViewHolder;
+
+import com.android.settings.R;
+
+import com.google.zxing.BarcodeFormat;   
+import com.google.zxing.EncodeHintType;   
+import com.google.zxing.MultiFormatWriter;   
+import com.google.zxing.WriterException;   
+import com.google.zxing.common.BitMatrix;
+
+import com.pos.sdk.accessory.PosAccessoryManager;
+import com.pos.sdk.printer.PosPrinter;
+
+import java.lang.StringBuilder;
+import java.util.Hashtable;
+
+public class QRDeviceInfoPreference extends Preference {
+
+    private static final String TAG = "QRDeviceInfoPreference";
+
+    private static final int BLACK = 0xff000000;
+
+    private static final int PRINTER_ERROR_NO_PAPER = 1;
+    private static final int PRINTER_ERROR_OVER_HEAT = 2;
+    private static final int PRINTER_STATE_CHANGED = 3;
+    private static final int PRINTER_ERROR_NO_BATTERY = 4;
+
+    private Context mContext;
+    private Bitmap qrCodeBitmap;
+    private Button mPrintBtn;
+    private ImageView qrImgImageView;
+    private PosPrinter mPrinter;
+    private TelephonyManager mTelephonyManager;
+    private String mSpVersion, mDsn, mPsn;
+
+    private PosPrinter.EventListener mListener = new PosPrinter.EventListener() {
+        @Override
+        public void onInfo(PosPrinter printer, int what, int extra) {
+            Log.i(TAG, "onInfo: what= " + what + ", extra= " + extra);
+            if (what == PosPrinter.PRINTER_INFO_STATE_CHANGED) {
+                mHandler.sendMessage(mHandler.obtainMessage(PRINTER_STATE_CHANGED, extra, 0));
+            }
+        }
+
+        @Override
+        public void onError(PosPrinter printer, int what, int extra) {
+            switch(what) {
+                case PosPrinter.PRINTER_ERROR_NO_PAPER:
+                    mHandler.sendEmptyMessage(PRINTER_ERROR_NO_PAPER);
+                    break;
+                case PosPrinter.PRINTER_ERROR_OVER_HEAT:
+                    Log.e(TAG, "PRINTER_ERROR_OVER_HEAT");
+                    mHandler.sendEmptyMessage(PRINTER_ERROR_OVER_HEAT);
+                    break;
+                case PosPrinter.PRINTER_ERROR_STATE:
+                    Log.e(TAG, "PRINTER_ERROR_STATE");
+                    if (extra == PosPrinter.PRINTER_STATE_NO_BATTERY) {
+                        mHandler.sendEmptyMessage(PRINTER_ERROR_NO_BATTERY);
+                    }
+                    break;
+                default:
+                    Log.e(TAG, "PRINTER_ERROR: " + what);
+                    break;
+            }
+        }
+
+        @Override
+        public void onCursorChanged(PosPrinter printer, int x, int y, int lastX, int lastY) {
+            Log.d(TAG, "onCursorChanged: x= " + x + ", y= " + ", lastX= " + ", lastY=" + lastY);
+        }
+    };
+
+    private Handler mHandler = new Handler() {
+        public void handleMessage(Message msg) {
+            super.handleMessage(msg);
+            switch (msg.what) {
+                case PRINTER_ERROR_NO_PAPER: {
+                    AlertDialog.Builder dlg = new AlertDialog.Builder(mContext);
+                    dlg.setTitle(R.string.reload_paper);
+                    dlg.setCancelable(false);
+                    dlg.setPositiveButton(R.string.been_installed_paper, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(false);
+                            mHandler.post(mPrinterRunnable);
+                        }
+                    });
+                    dlg.setNegativeButton(R.string.dlg_cancel, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(true);
+                        }
+                    });
+                    dlg.show();
+                    break;
+                }
+                case PRINTER_ERROR_OVER_HEAT: {
+                    AlertDialog.Builder dlg = new AlertDialog.Builder(mContext);
+                    dlg.setTitle(R.string.printer_overheat);
+                    dlg.setMessage(R.string.printer_overheat_summary);
+                    dlg.setCancelable(false);
+                    dlg.setPositiveButton(R.string.ok, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(true);
+                        }
+                    });
+                    dlg.show();
+                    break;
+                }
+                case PRINTER_ERROR_NO_BATTERY: {
+                    AlertDialog.Builder dlg = new AlertDialog.Builder(mContext);
+                    dlg.setTitle(R.string.printer_no_battery);
+                    dlg.setMessage(R.string.printer_no_battery_summary);
+                    dlg.setCancelable(false);
+                    dlg.setPositiveButton(R.string.ok, new DialogInterface.OnClickListener(){
+                        @Override
+                        public void onClick(DialogInterface dialog, int whichButton) {
+                            mPrintBtn.setEnabled(true);
+                        }
+                    });
+                    dlg.show();
+                    break;
+                }
+                case PRINTER_STATE_CHANGED:
+                    if (msg.arg1 == PosPrinter.PRINTER_STATE_PRINTING) {
+                        mPrintBtn.setEnabled(false);
+                    } else {
+                        mPrintBtn.setEnabled(true);
+                    }
+                    break;
+                default:
+                    break;
+            }
+        }
+    };
+
+    public QRDeviceInfoPreference(Context context, AttributeSet attributeSet) {
+        this(context, attributeSet, 0);
+    }
+
+    public QRDeviceInfoPreference(Context context, AttributeSet attributeSet, int paramInt) {
+        super(context, attributeSet, paramInt);
+        setLayoutResource(R.layout.qr_device_info_preference);
+        mContext = context;
+        mTelephonyManager = (TelephonyManager) context.getSystemService(Context.TELEPHONY_SERVICE);
+        mDsn = mTelephonyManager.getDsn(mContext);
+        mPsn = mTelephonyManager.getPsn(mContext);
+        mSpVersion = PosAccessoryManager.getDefault().getSpVersion();
+
+        int ret = -1;
+        int numOfPrinter = PosPrinter.getNumberOfPrinters();
+        if (numOfPrinter > 0) {
+            mPrinter = PosPrinter.open();
+            if (mPrinter != null) {
+                mPrinter.setOnEventListener(mListener);
+                ret = 0;
+            }
+        }
+        if (ret == -1) {
+            Log.d(TAG, "no printer");
+        }
+    }
+
+    @Override
+    public void onBindViewHolder(PreferenceViewHolder view) {
+        super.onBindViewHolder(view);
+
+        mPrintBtn = (Button) view.findViewById(R.id.print_button);
+        mPrintBtn.setOnClickListener(new View.OnClickListener() {
+            @Override
+            public void onClick(View v) {
+                mHandler.post(mPrinterRunnable);
+            }
+        });
+
+        qrImgImageView = (ImageView) view.findViewById(R.id.iv_qr_image);
+        qrImgImageView.setEnabled(false);
+        qrImgImageView.setOnClickListener(new View.OnClickListener() {
+            @Override
+            public void onClick(View v) {
+                showDialog();
+            }
+        });
+        qrImgImageView.setOnLongClickListener(new View.OnLongClickListener() {
+            @Override
+            public boolean onLongClick(View v) {
+                if (qrCodeBitmap != null) {
+                    print(qrCodeBitmap);
+                }
+                return true;
+            }
+        });
+
+        try {
+            String contentString = getDeviceInfo();
+            if (!TextUtils.isEmpty(contentString)) {
+                qrCodeBitmap = createQRCode(contentString, 350);
+                qrImgImageView.setImageBitmap(qrCodeBitmap);
+            }
+        } catch (WriterException e) {
+            // TODO Auto-generated catch block
+            e.printStackTrace();
+        }
+    }
+
+    private Bitmap createQRCode(String str,int widthAndHeight) throws WriterException {
+        Hashtable<EncodeHintType, String> hints = new Hashtable<EncodeHintType, String>();
+        hints.put(EncodeHintType.CHARACTER_SET, "utf-8");
+        BitMatrix matrix = new MultiFormatWriter().encode(str,
+                BarcodeFormat.QR_CODE, widthAndHeight, widthAndHeight);
+        int width = matrix.getWidth();
+        int height = matrix.getHeight();
+        int[] pixels = new int[width * height];
+
+        for (int y = 0; y < height; y++) {
+            for (int x = 0; x < width; x++) {
+                if (matrix.get(x, y)) {
+                    pixels[y * width + x] = BLACK;
+                }
+            }
+        }
+        Bitmap bitmap = Bitmap.createBitmap(width, height,
+                Bitmap.Config.ARGB_8888);
+        bitmap.setPixels(pixels, 0, width, 0, 0, width, height);
+
+        // add logo
+        Bitmap brandBitmap = BitmapFactory.decodeResource(mContext.getResources(), R.mipmap.company_logo);
+        if (brandBitmap != null) {
+            bitmap = addLogoToQRCode(bitmap, brandBitmap);
+        }
+        return bitmap;
+    }
+
+    private Bitmap addLogoToQRCode(Bitmap qrCodeBitmap, Bitmap logoBitmap) {
+        if (qrCodeBitmap == null) {
+            Log.d(TAG, "addLogoToQRCode::qrCodeBitmap is null");
+            return null;
+        }
+
+        if (logoBitmap == null) {
+            Log.d(TAG, "addLogoToQRCode::logoBitmap is null");
+            return qrCodeBitmap;
+        }
+
+        int qrWidth = qrCodeBitmap.getWidth();
+        int qrHeight = qrCodeBitmap.getHeight();
+        int logoWidth = logoBitmap.getWidth();
+        int logoHeight = logoBitmap.getHeight();
+
+        if (qrWidth == 0 || qrHeight == 0) {
+            Log.d(TAG, "addLogoToQRCode::qrCodeBitmap size is wrong");
+            return null;
+        }
+
+        if (logoWidth == 0 || logoHeight == 0) {
+            Log.d(TAG, "addLogoToQRCode::logoBitmap size is wrong");
+            return qrCodeBitmap;
+        }
+
+        float scaleFactor = qrWidth * 1.0f / 5 / logoWidth;
+        Bitmap resultBitmap = Bitmap.createBitmap(qrWidth, qrHeight, qrCodeBitmap.getConfig());
+        Canvas canvas = new Canvas(resultBitmap);
+        canvas.drawBitmap(qrCodeBitmap, 0, 0, null);
+
+        canvas.scale(scaleFactor, scaleFactor, qrWidth / 2, qrHeight / 2);
+        canvas.drawBitmap(logoBitmap, (qrWidth - logoWidth) / 2, (qrHeight - logoHeight) / 2, null);
+        canvas.save(Canvas.ALL_SAVE_FLAG);
+        canvas.restore();
+
+        return resultBitmap;
+    }
+
+    private void print(Bitmap bitmap) {
+        int num = PosPrinter.getNumberOfPrinters();
+        if (num > 0) {
+            mPrinter = PosPrinter.open();
+            mPrinter.printBitmap(bitmap);
+        }
+    }
+
+    private void showDialog() {
+        AlertDialog.Builder builder = new AlertDialog.Builder(mContext);
+        final View dialogView = LayoutInflater.from(mContext).inflate(R.layout.qr_device_info_preference, null);
+        ImageView mImageView = (ImageView) dialogView.findViewById(R.id.iv_qr_image);
+        mImageView.setImageBitmap(qrCodeBitmap);
+        builder.setView(dialogView);
+        builder.setPositiveButton(R.string.print_settings, new DialogInterface.OnClickListener() { 
+            @Override 
+            public void onClick(DialogInterface dialog, int which) {
+                if (qrCodeBitmap != null) {
+                    print(qrCodeBitmap);
+                }
+            } 
+        });
+        builder.show();
+    }
+
+    private String getDeviceInfo() {
+        String model = "Model:" + Build.MODEL;
+        String dsn = "DSN:" + mDsn;
+        String psn = "PSN:" + mPsn;
+        String ap_version = "AP:" + SystemProperties.get("ro.build.sw.version");
+        String sp_version = "SP:" + mSpVersion;
+        String modem_version = "Modem:" + Build.getRadioVersion();
+        StringBuilder sb = new StringBuilder();
+        sb.append(model + "\n");
+        sb.append(dsn + "\n");
+        sb.append(psn + "\n");
+        sb.append(ap_version + "\n");
+        sb.append(sp_version + "\n");
+        sb.append(modem_version + "\n");
+        return sb.toString();
+    }
+
+    private Runnable mPrinterRunnable = new Runnable(){
+        @Override
+        public void run() {
+            mPrinter.cleanCache();
+            PosPrinter.Parameters param = mPrinter.getParameters();
+
+            param.setPrintAlign(param.ALIGH_CENTER);
+            mPrinter.setParameters(param);
+            Bitmap brandBitmap = BitmapFactory.decodeResource(mContext.getResources(), R.mipmap.company_logo);
+            if (brandBitmap != null) {
+                mPrinter.addBitmapToCache(brandBitmap);
+            }
+
+            param.setFontSize(30);
+            param.setPrintAlign(param.ALIGH_CENTER);
+            param.setFontName(Environment.getRootDirectory().toString() + "/fonts/DroidSansMono.ttf");
+            param.setFontEffet(param.EFFECT_BOLD);
+            mPrinter.setParameters(param);
+            mPrinter.addTextToCurCache("---" + mContext.getResources().getString(R.string.device_info) + "---");
+            
+            param.setFontSize(24);
+            param.setPrintAlign(param.ALIGH_LEFT);
+            param.setFontName(Environment.getRootDirectory().toString() + "/fonts/Roboto-Italic.ttf");
+            param.setFontEffet(param.EFFECT_NONE);
+            mPrinter.setParameters(param);
+            mPrinter.addTextToCurCache("Model: " + Build.MODEL);
+            mPrinter.addTextToCurCache("AP: " + SystemProperties.get("ro.build.sw.version"));
+            mPrinter.addTextToCurCache("SP: " + mSpVersion);
+            mPrinter.addTextToCurCache("DSN: " + mDsn);
+            mPrinter.addTextToCurCache("PSN: " + mPsn);
+            mPrinter.addTextToCurCache("Modem: " + Build.getRadioVersion());
+
+            param.setFontSize(18);
+            param.setPrintAlign(param.ALIGH_CENTER);
+            param.setFontName(Environment.getRootDirectory().toString() + "/fonts/DroidSansMono.ttf");
+            param.setFontEffet(param.EFFECT_LEAN);
+            mPrinter.setParameters(param);
+            mPrinter.addTextToCurCache("-----" + mContext.getResources().getString(R.string.website) + "-----");
+
+            if (qrCodeBitmap != null) {
+                mPrinter.addBitmapToCache(qrCodeBitmap);
+            }
+            mPrinter.addTextToCurCache("\n\n\n");
+            mPrinter.print();
+        }
+    };
+}

4. 控制preference是否显示

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/config.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/config.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/config.xml	(版本 347)
@@ -649,4 +649,6 @@+
+    <bool name="config_show_qr_info">true</bool></resources>
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreferenceController.java
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreferenceController.java	(不存在的)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/src/com/android/settings/deviceinfo/QRDeviceInfoPreferenceController.java	(版本 347)
@@ -0,0 +1,35 @@
+package com.android.settings.deviceinfo;
+
+import android.content.Context;
+import androidx.preference.PreferenceScreen;
+
+import com.android.settings.R;
+import com.android.settings.core.BasePreferenceController;
+import com.android.settings.widget.QRDeviceInfoPreference;
+
+public class QRDeviceInfoPreferenceController extends BasePreferenceController {
+    private static final String PREF_KEY = "qr_info";
+    private QRDeviceInfoPreference mPreference;
+
+    public QRDeviceInfoPreferenceController(Context context) {
+        super(context, PREF_KEY);
+    }
+
+    @Override
+    public void displayPreference(PreferenceScreen screen) {
+        super.displayPreference(screen);
+        mPreference = (QRDeviceInfoPreference) screen.findPreference(PREF_KEY);
+    }
+
+    @Override
+    public int getAvailabilityStatus() {
+        return mContext.getResources()
+                .getBoolean(R.bool.config_show_qr_info)
+                ? AVAILABLE : DISABLED_FOR_USER;
+    }
+
+    @Override
+    public String getPreferenceKey() {
+        return PREF_KEY;
+    }
+}

5. 相关字串

Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/strings.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/strings.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values/strings.xml	(版本 347)
@@ -364,4 +364,15 @@
+
+    <string name="print">print</string>
+    <string name="reload_paper">reload paper</string>
+    <string name="dlg_cancel">cancel</string>
+    <string name="been_installed_paper">paper ready</string>
+    <string name="printer_overheat">printer overheat</string>
+    <string name="printer_overheat_summary">printer overheat, please try again later</string>
+    <string name="printer_no_battery">printer no battery</string>
+    <string name="printer_no_battery_summary">no battery, please install the battery and try again</string>
+    <string name="device_info">DEVICE INFO</string>
+    <string name="website">www.xxx.cn</string></resources>
Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values-zh-rCN/strings.xml
===================================================================
--- vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values-zh-rCN/strings.xml	(版本 346)
+++ vendor/mediatek/proprietary/packages/apps/MtkSettings/res/values-zh-rCN/strings.xml	(版本 347)
@@ -386,4 +386,15 @@
+    <string name="print">打印</string>
+    <string name="reload_paper">请重新装纸</string>
+    <string name="dlg_cancel">"取消"</string>
+    <string name="been_installed_paper">已经装好纸</string>
+    <string name="printer_overheat">温度过热</string>
+    <string name="printer_overheat_summary">打印机温度过热,请稍候再试</string>
+    <string name="printer_no_battery">没有电池</string>
+    <string name="printer_no_battery_summary">未安装电池,请安装电池再试</string>
+    <string name="no_printer">没有找到打印机</string>
+    <string name="no_printer_summary">没有找到打印机,请检查打印硬件是否正常</string>
+    <string name="device_info">设备信息</string></resources>

相关文章:

android设置添加设备QR码信息

摘要&#xff1a;客户衍生需求&#xff0c;通过扫QR码快速获取设备基础信息&#xff0c;并且基于POS SDK进行打印。 1. 定位至device info的xml添加相关perference Index: vendor/mediatek/proprietary/packages/apps/MtkSettings/res/xml/my_device_info.xml--- vendor/medi…...

进程状态

目录 1.进程排队 硬件的队列 进程排队 2.进程的三大状态 什么是状态 运行状态 阻塞状态 挂起状态 3.Linux系统中的进程状态 4.僵尸状态 5.孤儿进程 1.进程排队 硬件的队列 计算机是由很多硬件组成的&#xff0c;操作系统为了管理这些硬件&#xff0c;通常需要为这…...

【linux学习指南】模拟线程封装与智能指针shared_ptr

文章目录 &#x1f4dd;线程封装&#x1f309; Thread.hpp&#x1f309; Makefile &#x1f320;线程封装第一版&#x1f309; Makefile:&#x1f309;Main.cc&#x1f309; Thread.hpp: &#x1f320;线程封装第二版&#x1f309; Thread.hpp:&#x1f309; Main.cc &#x1f…...

智慧物流新引擎:ARM架构工控机在自动化生产线中的应用

工业自动化程度的不断提升&#xff0c;对高性能、低功耗和高可靠性的计算设备需求日益增长。ARM架构工控机因其独特的优势&#xff0c;在多个工业领域得到了广泛应用。本文将深入探讨ARM架构工控机的特点及其在具体工业场景中的应用。 ARM架构工控机的主要优势 高效能与低功耗…...

OpenGL的基础光照知识

光照模型 常见的光照模型&#xff1a;ADS模型 A&#xff1a;环境光反射&#xff08;ambient reflection&#xff09;&#xff1a;模拟低级光照&#xff0c;影响场景中的所有物体。D&#xff1a;漫反射&#xff08;diffuse reflection&#xff09;&#xff1a;根据光线的入射角…...

centos 10 离线安装dnf 和 设置dnf镜像源

离线安装dnf可用kimi搜索, centos 使用curl 下载dnf 的rpm包 mkdir ~/dnf_packages cd ~/dnf_packages# CentOS 7 示例 curl -O http://springdale.math.ias.edu/data/puias/unsupported/7/x86_64/dnf-0.6.4-2.sdl7.noarch.rpm curl -O http://springdale.math.ias.edu/data/pu…...

redis 缓存击穿问题与解决方案

前言1. 什么是缓存击穿?2. 如何解决缓存击穿?怎么做?方案1: 定时刷新方案2: 自动续期方案3: 定时续期 如何选? 前言 当我们使用redis做缓存的时候,查询流程一般是先查询redis,如果redis未命中,再查询MySQL,将MySQL查询的数据同步到redis(回源),最后返回数据 流程图 为什…...

Linux下的进程切换与调度

目录 1.进程的优先级 优先级是什么 Linux下优先级的具体做法 优先级的调整为什么要受限 2.Linux下的进程切换 3.Linux下进程的调度 1.进程的优先级 我们在使用计算机的时候&#xff0c;通常会启动多个程序&#xff0c;这些程序最后都会变成进程&#xff0c;但是我们的硬…...

开源模型应用落地-Qwen1.5-MoE-A2.7B-Chat与vllm实现推理加速的正确姿势(一)

一、前言 在人工智能技术蓬勃发展的当下,大语言模型的性能与应用不断突破边界,为我们带来前所未有的体验。Qwen1.5-MoE-A2.7B-Chat 作为一款备受瞩目的大语言模型,以其独特的架构和强大的能力,在自然语言处理领域崭露头角。而 vllm 作为高效的推理库,为模型的部署与推理提…...

阿里云IOT设备管理

本文主要介绍了阿里云IOT设备管理的基本概念、功能特点以及应用场景。阐述了如何利用阿里云IOT平台实现设备的连接、监控和控制&#xff0c;以及如何借助其丰富的数据分析功能提升设备管理效率。 一、IOT工作原理 二、创建模拟设备 1.创建产品 2.物模型 3.设备 4.设备数据上报…...

图像处理技术和应用

图像处理技术是一种依托计算机和相关算法&#xff0c;对图像进行深度处理、分析及改变的技术。主要包括图像数字化、图像增强和复原、图像数据编码、图像分割和图像识别等。它不仅能够从静态图像中提取关键信息&#xff0c;还能改变图像的外观或特征&#xff0c;并进一步检测、…...

格式化字符串漏洞详解

一、漏洞原理 格式化字符串漏洞&#xff08;Format String Vulnerability&#xff09;是由于程序使用用户可控的输入作为格式化字符串参数&#xff08;如 printf、sprintf 等函数&#xff09;时未正确过滤导致的漏洞。攻击者可通过构造特殊格式字符串实现以下操作&#xff1a;…...

java项目之基于web的中国古诗词的设计与实现源码(ssm+mysql)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的基于web的中国古诗词的设计与实现。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 基于web的中国…...

网络初识-

网络的相关概念 一、局域网和广域网 将各种计算机、外部设备等相互连接起来&#xff0c;实现在这个范围内数据通信和资源共享的计算机网络。它的覆盖范围通常在几百米到几公里之内。例如&#xff0c;一个小型企业的办公室&#xff0c;通过交换机将多台电脑连接在一起&#xf…...

AOS安装及操作演示

文章目录 一、安装node1.1 在 macOS 上管理 Node版本1.1.1 安装 nvm1.1.2 验证 nvm 是否安装成功1.1.3 使用 nvm 安装/切换 Node.js 版本1.1.4 卸载 Node.js 版本 1.2 在 windows 上管理 Node版本1.2.1 安装 nvm-windows1.2.2 安装 Node.js 版本1.2.3 切换 Node.js 版本1.2.4 卸…...

vue学习8

1.pinia&#xff08;更优&#xff09; 是vue最新的状态管理工具&#xff0c;是vuex的替代品 pinia&#xff1a; state actions(支持异步&#xff0c;可以直接修改state) getters 优点&#xff1a; 提供更加简单的API(去掉了mutation)提供符合&#xff0c;组合式的API语法(和v…...

【竞技宝】电竞世界杯:无畏契约首次入选正式项目!

北京时间2月12日&#xff0c;电竞世界杯基金会&#xff08;EWCF&#xff09;与知名游戏开发商拳头游戏&#xff08;Riot Games&#xff09;在近日共同宣布达成三年合作伙伴关系。同时&#xff0c;三大顶级电竞项目——《英雄联盟》《英雄联盟&#xff1a;云顶之弈》&#xff08…...

Bigemap Pro地图配置文件包

配置文件获取 配置文件下载后&#xff0c;直接拖入软件中自动识别导入图源&#xff0c;一键完成加载。...

有哪些免费的SEO软件优化工具

随着2025年互联网的不断发展&#xff0c;越来越多的企业意识到在数字营销中&#xff0c;网站的曝光度和排名至关重要。无论是想要提高品牌知名度&#xff0c;还是想要通过在线销售增加收益&#xff0c;SEO&#xff08;搜索引擎优化&#xff09;都是一项不可忽视的关键策略。而要…...

第二天:工具的使用

每天上午9点左右更新一到两篇文章到专栏《Python爬虫训练营》中&#xff0c;对于爬虫有兴趣的伙伴可以订阅专栏一起学习&#xff0c;完全免费。 键盘为桨&#xff0c;代码作帆。这趟为期30天左右的Python爬虫特训即将启航&#xff0c;每日解锁新海域&#xff1a;从Requests库的…...

分享在职同时准备系统分析师和教资考试的时间安排

&#xff08;在职、时间有限、同时备考系统分析师考试和小学信息技术教资面试&#xff09;&#xff0c;以下是详细的备考计划&#xff0c;确保计划的可行性和通过性。 一、总体安排 时间分配&#xff1a; 每周周末&#xff08;2天&#xff09;用于系统分析师考试备考。工作日晚…...

从Word里面用VBA调用NVIDIA的免费DeepSeekR1

看上去能用而已。 选中的文字作为输入&#xff0c;运行对应的宏即可&#xff1b;会先MSGBOX提示一下&#xff0c;然后相关内容追加到word文档中。 需要自己注册生成好用的apikey Option ExplicitSub DeepSeek()Dim selectedText As StringDim apiKey As StringDim response A…...

3.2 > Bash

概览 在上一节中我们了解了关于 Shell 的执行流程&#xff0c;知道了在 Linux 环境中一般有哪些常用的 Shell。而在本节中&#xff0c;将会学习到 Linux 中最常见的一个 Shell —— Bash&#xff0c;了解到 bash 的相关知识和用法。 本节目录 概览相关知识bash 命令提示符bas…...

游戏引擎学习第100天

仓库:https://gitee.com/mrxiao_com/2d_game_2 昨天的回顾 今天的工作重点是继续进行反射计算的实现。昨天&#xff0c;我们开始了反射和环境贴图的工作&#xff0c;成功地根据法线显示了反射效果。然而&#xff0c;我们还没有实现反射向量的计算&#xff0c;导致反射交点的代…...

新一代SCADA: 宏集Panorama Suite 2025 正式发布,提供更灵活、符合人体工学且安全的应用体验

宏集科技宣布正式推出全新Panorama Suite 2025 SCADA软件&#xff01;全新版本标志着 Panorama Suite的一个重要里程碑&#xff0c;代表了从 Panorama Suite 2022 开始并跨越三个版本&#xff08;2022、2023、2025&#xff09;的开发过程的顶峰。 此次重大发布集中在六个核心主…...

Visual Studio 进行单元测试【入门】

摘要&#xff1a;在软件开发中&#xff0c;单元测试是一种重要的实践&#xff0c;通过验证代码的正确性&#xff0c;帮助开发者提高代码质量。本文将介绍如何在VisualStudio中进行单元测试&#xff0c;包括创建测试项目、编写测试代码、运行测试以及查看结果。 1. 什么是单元测…...

Notepad++ 中删除所有以 “pdf“ 结尾的行

Notepad 中删除所有以 “pdf” 结尾的行 操作步骤 1.打开文件&#xff1a; 在 Notepad 中打开你需要处理的文本文件。 2.打开查找和替换对话框&#xff1a; 按快捷键 Ctrl F&#xff0c;打开“查找和替换”对话框。 3.启用正则表达式模式&#xff1a; 在对话框的底部&#xf…...

Java 使用腾讯翻译 API 实现含 HTML 标签文本,json值,精准翻译工具

注意&#xff1a;需搭配标题二的腾讯翻译工具使用 一-1、翻译标签文本工具 package org.springblade.common.utils;import java.util.ArrayList; import java.util.List; import java.util.regex.Matcher; import java.util.regex.Pattern;public class TencentTranslationFor…...

DeepSeek R1+Open WebUI +SearXNG 本地化部署与联网功能

GitHub - searxng/searxng-docker: The docker-compose files for setting up a SearXNG instance with docker....

数据科学之数据管理|NumPy数据管

一、Numpy介绍 (一) 什么是numpy NumPy是Python中科学计算的基础包。它是一个Python库,提供多维数组对象,各种派生对象(如掩码数组和矩阵),以及用于数组快速操作的各种API,有包括数学、逻辑、形状操作、排序、选择、输入输出、离散傅立叶变换、基本线性代数,基本统计运…...

零基础玩转 DeepSeek API实战教程

大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于大模型算法的研究与应用。曾担任百度千帆大模型比赛、BPAA算法大赛评委,编写微软OpenAI考试认证指导手册。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。授权多项发明专利。对机器学…...

【GPIO】5.理解保护二极管在GPIO过电压保护中的作用

在电子电路设计中&#xff0c;保护二极管是常见的保护元件&#xff0c;用于防止过电压对敏感电路的损害。本文将探讨当GPIO输入电压大于3.3V时&#xff0c;保护二极管如何工作&#xff0c;并解释为什么大部分过电压引起的电流会通过二极管流向VDD而不是流入内部电路。 1.背景 …...

2.5 模块化迁移策略:从传统项目到模块化系统

模块化迁移策略&#xff1a;从传统项目到模块化系统 将传统 Java 项目迁移至 JDK 9 模块化系统是一项系统性工程&#xff0c;需分阶段实施以降低风险。以下是详细的迁移策略、工具使用和实战示例。 1. 迁移阶段划分 阶段目标关键操作阶段1&#xff1a;兼容性验证确保项目能在…...

Tortoise Git

TortoiseGit 是一个 Windows Shell 与 Git 的接口&#xff0c;它提供了文件状态的覆盖图标&#xff0c;强大的 Git 上下文菜单等。你可以在官方网站 (tortoisegit.org) 轻松使用安装程序进行下载。TortoiseGit 的当前稳定版本是 2.14.0 &#xff0c;根据你的机器配置&#xff0…...

Maven Spring框架依赖包

Maven中添加Spring框架依赖包 Spring核心工具包SpringJDBCSpring配置文件头信息 Spring核心工具包 在pom.xml文件中添加 <!-- Spring的核心工具包--><dependencies><dependency><groupId>org.springframework</groupId><artifactId>spr…...

【Cocos TypeScript 零基础 15.1】

目录 见缝插针UI脚本针脚本球脚本心得_旋转心得_更改父节点心得_缓动动画成品展示图 见缝插针 本人只是看了老师的大纲,中途不明白不会的时候再去看的视频 所以代码可能与老师代码有出入 SIKI_学院_点击跳转 UI脚本 import { _decorator, Camera, color, Component, directo…...

Linux库制作与原理:【静态库】【动态库】【目标文件】【ELF文件】【ELF从形成到假造轮廓】【理解链接和加载】

目录 一.什么是库 二.静态库 2.1创建静态库 我们在之前的路径下新建lib使用我们自己的库 2.2 使用makefile生成静态库 三.动态库 3.1动态库生成 3.2动态库使用 3.3库运行搜索路径 四.目标文件 五.ELF文件 六.ELF从形成到加载轮廓 6.1ELF形成可执行 6.2 ELF可执行文…...

中间件-redis-(ubantu)

1、安装依赖包 sudo apt-get update sudo apt-get install redis 一旦安装完成&#xff0c;Redis 服务将会自动启动。想要检查服务的状态&#xff0c;输入下面的命令&#xff1a; rootvims:/etc/redis# sudo systemctl status redis-server ● redis-server.service - Adva…...

ubuntu20.04+ROS+Gazebo+px4+QGC+MAVROS

目录 前言 一、安装ROS 二、安装PX4 编译 三、QGC安装 四、安装MAVROS 命令记得加sudo&#xff01; 前言 在安装ubuntu20.04ROSGazebopx4QGCMAVROS时&#xff0c;参考了很多网上的资料&#xff0c;总结一个较为顺利的流程。 官方指南PX4 自动驾驶仪用户指南 | PX4 Gui…...

基于 openEuler 构建 LVS-DR 群集(同网段)。

一、LVS相关原理 1.LVS简介 LVS是Linux Virtual Server的简称&#xff0c;也就是Linux虚拟服务器, 是一个由章文嵩博士发起的自由软件项 目&#xff0c;它的官方站点是www.linuxvirtualserver.org。现在LVS已经是 Linux标准内核的一部分&#xff0c;在 Linux2.4内核以前&…...

计算机毕业设计PySpark+Hadoop+Hive机票预测 飞机票航班数据分析可视化大屏 航班预测系统 机票爬虫 飞机票推荐系统 大数据毕业设计

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…...

【设计模式】【行为型模式】观察者模式(Observer)

&#x1f44b;hi&#xff0c;我不是一名外包公司的员工&#xff0c;也不会偷吃茶水间的零食&#xff0c;我的梦想是能写高端CRUD &#x1f525; 2025本人正在沉淀中… 博客更新速度 &#x1f4eb; 欢迎V&#xff1a; flzjcsg2&#xff0c;我们共同讨论Java深渊的奥秘 &#x1f…...

机器学习: 逻辑回归

概念与定义 逻辑回归是一种用于分类问题的统计方法。它通过计算目标变量的概率来预测类别归属,并假设数据服从伯努利分布(二分类)或多项式分布(多分类)。逻辑回归模型输出的是概率值,通常使用sigmoid函数将线性组合映射到0和1之间。 1. 概念 逻辑回归用于解决分类问题…...

域名解析—互联网世界的导航系统

在互联网的世界里&#xff0c;每个网站都像一座“城市”&#xff0c;而用户要找到这些“城市”&#xff0c;必须依赖一套精准的导航系统——这就是域名解析。无论是浏览网页、发送邮件&#xff0c;还是使用移动应用&#xff0c;域名解析都在背后默默支撑着用户的每一次访问。本…...

PAT乙级真题 — 1080 MOOC期终成绩(java)【测试点3超时】

对于在中国大学MOOC&#xff08;http://www.icourse163.org/ &#xff09;学习“数据结构”课程的学生&#xff0c;想要获得一张合格证书&#xff0c;必须首先获得不少于200分的在线编程作业分&#xff0c;然后总评获得不少于60分&#xff08;满分100&#xff09;。总评成绩的计…...

【Prometheus】如何通过prometheus监控redis实时运行状态,并实现告警通知

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…...

ARM Cortex-M3/M4 权威指南 笔记【一】技术综述

一、Cortex-M3/M4 处理器的一般信息 1.1 处理器类型 ARM Cortex-M 为 32 位 RISC&#xff08;精简指令集&#xff09;处理器&#xff0c;其具有&#xff1a; 32位寄存器32位内部数据通路32位总线接口 除了 32 位数据&#xff0c;Cortex-M 处理器&#xff08;以及其他任何 A…...

【Qt】定期清理程序

在现有Qt程序中实现可配置日志保存天数的代码示例&#xff0c;分为界面修改、配置存储和核心逻辑三部分&#xff1a; // 1. 在配置文件&#xff08;如settings.h&#xff09;中添加保存天数的配置项 class Settings { public:int logRetentionDays() const {return m_settings…...

基于51单片机的门禁刷卡器proteus仿真

地址&#xff1a;https://pan.baidu.com/s/1j0KAmH5pVGWZWRpT6p5hBg 提取码&#xff1a;1234 仿真图&#xff1a; 芯片/模块的特点&#xff1a; AT89C52/AT89C51简介&#xff1a; AT89C52/AT89C51是一款经典的8位单片机&#xff0c;是意法半导体&#xff08;STMicroelectron…...

二、数据持久化篇(深度增强版)

二、数据持久化篇&#xff08;深度增强版&#xff09; 2.1 JDBC Template深度解析 架构设计思想 #mermaid-svg-y2IrKiVu2gzenoCB {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-y2IrKiVu2gzenoCB .error-icon{fil…...