Advertisement
nozeros

Stanislav Comendant NEA source code

Apr 2nd, 2025
513
0
Never
Not a member of Pastebin yet? Sign Up, it unlocks many cool features!
Python 23.44 KB | Source Code | 0 0
  1. ## Libraries ##
  2. import customtkinter
  3. from PIL import Image
  4. import webview
  5. import plotly.graph_objects as go
  6. import sys
  7.  
  8. import math
  9. import numpy as np
  10.  
  11. ## Functions ##
  12. def main():
  13.     def checkbox():    # Creates the mass and diameter input boxes if the checkbox is checked, otherwise it destroys them
  14.                        
  15.         if dragcheck.get():
  16.  
  17.             # Labels and input boxes for mass and diameter
  18.             global masstext, massinp, massinfo
  19.             masstext=customtkinter.CTkLabel(inputframe, fg_color="transparent", text="Enter mass:", text_color=labelcolour)
  20.             masstext.place(x=83, y=640)
  21.             massinp=customtkinter.CTkEntry(master=inputframe, placeholder_text="n/a")
  22.             massinp.pack(pady=50, padx=0)
  23.  
  24.             massinfo=customtkinter.CTkButton(inputframe, text="?",width=15,height=15, border_width=1, border_color="gray",command=lambda: info("Changes the mass of the projectile.\nTakes float values above 0 in kilograms (metric) or pounds (imperial).\nA projectile's mass affects its motion by reducing the \ndecelerating impact of air resistance."))
  25.             massinfo.place(y=640,x=265)
  26.  
  27.             global diametertext, diameterinp,diameterinfo
  28.             diametertext=customtkinter.CTkLabel(inputframe, fg_color="transparent",text="Enter diameter:", text_color=labelcolour)
  29.             diametertext.place(x=83, y=768)
  30.             diameterinp=customtkinter.CTkEntry(master=inputframe, placeholder_text="n/a")
  31.             diameterinp.pack(pady=50, padx=0)
  32.  
  33.             diameterinfo=customtkinter.CTkButton(inputframe, text="?",width=15,height=15, border_width=1, border_color="gray",command=lambda: info("Changes the diameter of the sphere.\nTakes float values above 0 in metres (metric) or feet (imperial).\nA higher diameter means a higher cross-sectional area, \nleading to a higher drag force and more deceleration. "))
  34.             diameterinfo.place(y=768,x=265)
  35.  
  36.             global dragcoeftext, dragcoefinp, dragcoefinfo
  37.             dragcoeftext=customtkinter.CTkLabel(inputframe, fg_color="transparent",text="Enter drag coefficient:", text_color=labelcolour)
  38.             dragcoeftext.place(x=83, y=896)
  39.             dragcoefinp=customtkinter.CTkEntry(master=inputframe, placeholder_text="Default: 0.47")
  40.             dragcoefinp.pack(pady=50, padx=0)
  41.  
  42.             dragcoefinfo=customtkinter.CTkButton(inputframe, text="?",width=15,height=15, border_width=1, border_color="gray",command=lambda: info("Changes the drag coefficient of the projectile.\nTakes float values above 0.\nA higher drag coefficient means a higher drag force and more deceleration.\n\nSome common drag coefficients are: (none are hollow)\nSphere: 0.47\n"
  43.             "Cube (face facing forward): 1.05\n Cube (corner facing forward): 0.8\nSemisphere (concave side): 0.42\nSemisphere (convex side): 1.17\nTetrahedron (pointy side): 0.50\nPlate: 1.17"))
  44.             dragcoefinfo.place(y=896,x=265)
  45.  
  46.         else:
  47.             if 'masstext' in globals():   # Destroys the mass and diameter input boxes if the checkbox is unchecked, and if they exist
  48.                 masstext.destroy()
  49.                 massinp.destroy()
  50.                 massinfo.destroy()
  51.  
  52.             if 'diametertext' in globals():
  53.                 diametertext.destroy()
  54.                 diameterinp.destroy()
  55.                 diameterinfo.destroy()
  56.  
  57.             if 'dragcoeftext' in globals():
  58.                 dragcoeftext.destroy()
  59.                 dragcoefinp.destroy()
  60.                 dragcoefinfo.destroy()
  61.            
  62.     def info(message):      # Creates an info window with a message
  63.        
  64.         global infowindow     # If an info window is already opened, the old one gets destroyed and a new one is created
  65.         try:
  66.             if 'normal' == infowindow.state():
  67.                 infowindow.destroy()
  68.         except:
  69.                 pass
  70.        
  71.         # Info window with its contents
  72.         infowindow=customtkinter.CTk(fg_color=backgroundcolour)
  73.         infowindow.geometry("500x300")
  74.         infowindow.resizable(False,False)
  75.         infowindow.title("?? Info ??")
  76.  
  77.         text=customtkinter.CTkLabel(infowindow,text=message,fg_color="transparent",text_color=labelcolour)
  78.         text.pack(pady=50,padx=10)
  79.  
  80.         mainwindow.protocol("WM_DELETE_WINDOW", lambda: close(infowindow))  # Closes the info window and the main window when the 'X' button on the main window is pressed
  81.  
  82.         infowindow.mainloop()
  83.    
  84.  
  85.  
  86.     global mainwindow,inputframe,dragcheck,angleinp,velocityinp,gravityinp,ystartinp,dragcoefinp
  87.  
  88.     customtkinter.set_appearance_mode(theme)
  89.     customtkinter.set_default_color_theme(accentcolour)
  90.     customtkinter.deactivate_automatic_dpi_awareness()
  91.  
  92.     # Main menu window
  93.     mainwindow = customtkinter.CTk(fg_color=backgroundcolour)
  94.     mainwindow.title("Projectile motion simulator")
  95.     mainwindow.geometry("345x1200")
  96.     mainwindow.resizable(False, False)
  97.  
  98.     # Main frame in the main menu
  99.     inputframe = customtkinter.CTkFrame(mainwindow,fg_color=framecolour)      
  100.     inputframe.pack(pady=(90, 20), padx=(20, 20), fill="both", expand=True,)
  101.  
  102.     # Buttons
  103.     runbutton = customtkinter.CTkButton(mainwindow, text="Run", width=120, height=40, border_width=1, border_color="gray", command=runsim)      
  104.     runbutton.place(x=41, y=21)
  105.  
  106.     settingpng = customtkinter.CTkImage(light_image=Image.open("settings.png"))                                                              
  107.     settingbutton = customtkinter.CTkButton(mainwindow, text="", image=settingpng, width=40, height=40, border_width=1, border_color="gray", command=settings)
  108.     settingbutton.place(x=255, y=21)
  109.  
  110.     # Labels and input boxes for each variable (airdrag, angle, velocity, gravity)
  111.     dragcheck = customtkinter.CTkCheckBox(inputframe, text="Air resistance", command=checkbox,text_color=labelcolour)
  112.     dragcheck.pack(pady=50)
  113.  
  114.     angletext = customtkinter.CTkLabel(inputframe, fg_color="transparent", text="Enter angle:", text_color=labelcolour)
  115.     angletext.place(x=83, y=128)
  116.     angleinp = customtkinter.CTkEntry(inputframe, placeholder_text="n/a")
  117.     angleinp.pack(pady=50, padx=0)
  118.  
  119.     angleinfo = customtkinter.CTkButton(inputframe, text="?", width=15, height=15, border_width=1, border_color="gray", command=lambda: info("Changes the angle with the ground at which the projectile is launched.\nTakes float values between 0-90 degrees, inclusive"))
  120.     angleinfo.place(y=136, x=265)
  121.  
  122.     velocitytext = customtkinter.CTkLabel(inputframe, fg_color="transparent", text="Enter initial velocity:", text_color=labelcolour)
  123.     velocitytext.place(x=83, y=256)
  124.     velocityinp = customtkinter.CTkEntry(inputframe, placeholder_text="n/a")
  125.     velocityinp.pack(pady=50, padx=0)
  126.  
  127.     velocityinfo = customtkinter.CTkButton(inputframe, text="?", width=15, height=15, border_width=1, border_color="gray", command=lambda: info("Changes the velocity at which the projectile is shot.\nTakes float values above 0 in metres per second (metric) or \nfeet per second (imperial)."))
  128.     velocityinfo.place(y=262, x=265)
  129.  
  130.     gravitytext = customtkinter.CTkLabel(inputframe, fg_color="transparent", text="Enter gravity:" ,text_color=labelcolour)
  131.     gravitytext.place(x=83, y=384)
  132.     if units=="Metric":
  133.         gravityinp = customtkinter.CTkEntry(master=inputframe, placeholder_text="Default: 9.81 m/s^2")
  134.     else:
  135.         gravityinp = customtkinter.CTkEntry(master=inputframe, placeholder_text="Default: 32.19 ft/s^2")
  136.     gravityinp.pack(pady=50, padx=0)
  137.  
  138.     gravityinfo = customtkinter.CTkButton(inputframe, text="?", width=15, height=15, border_width=1, border_color="gray", command=lambda: info("Changes the acceleration of the projectile to the ground due to gravity.\nTakes float values above 0 in metres per second squared (metric) \nor feet per second squared (imperial)."))
  139.     gravityinfo.place(y=388, x=265)
  140.  
  141.     ystarttext= customtkinter.CTkLabel(inputframe, fg_color="transparent", text="Enter starting height:", text_color=labelcolour)
  142.     ystarttext.place(x=83, y=512)
  143.     if units=="Metric":
  144.         ystartinp = customtkinter.CTkEntry(inputframe, placeholder_text="Default: 0 m")
  145.     else:
  146.         ystartinp = customtkinter.CTkEntry(inputframe, placeholder_text="Default: 0 ft")
  147.     ystartinp.pack(pady=50, padx=0)
  148.  
  149.     ystartinfo=customtkinter.CTkButton(inputframe, text="?", width=15, height=15, border_width=1, border_color="gray", command=lambda: info("Changes the height at which the projectile starts.\nTakes float values above 0 in metres (metric) or feet (imperial)."))
  150.     ystartinfo.place(y=514, x=265)
  151.    
  152.     tooltip = customtkinter.CTkButton(mainwindow, text="?", width=28, height=28, border_width=1, border_color="gray", command=lambda: info("Some helpful keybinds for quicker acess:\n\nPress 'Enter','Space' or 'R' to run the simulation.\nPress 'S' to open settings.\nPress 'Esc' to exit the program."))
  153.     tooltip.place(y=21, x=300)
  154.  
  155.  
  156.     # Keybinds
  157.     mainwindow.bind("<Return>", runsim)  # Return=Enter
  158.     mainwindow.bind("<space>", runsim)
  159.     mainwindow.bind("<r>", runsim)
  160.     mainwindow.bind("<s>", settings)
  161.     mainwindow.bind("<Escape>", lambda e: sys.exit()) # Escape closes the window
  162.  
  163.  
  164.     mainwindow.mainloop()
  165.  
  166. def runsim(event=None):
  167.     def warning(message):       # Creates a warning window with a message
  168.        
  169.         global warningwindow     # If a warning window is already opened, the old one gets destroyed and a new one is created
  170.         try:
  171.             if 'normal' == warningwindow.state():
  172.                 warningwindow.destroy()
  173.         except:
  174.                 pass
  175.  
  176.         # Warning window with its contents
  177.         warningwindow=customtkinter.CTk()
  178.         warningwindow.geometry("500x300")
  179.         warningwindow.resizable(False,False)
  180.         warningwindow.title("!! Warning !!")
  181.  
  182.         warningtext=customtkinter.CTkLabel(warningwindow, fg_color="transparent", text=message)
  183.         warningtext.pack(pady=100,padx=10)
  184.  
  185.         mainwindow.protocol("WM_DELETE_WINDOW", lambda: close(warningwindow))  # Closes the warning window and the main window when the 'X' button on the main window is pressed
  186.  
  187.         warningwindow.mainloop()
  188.        
  189.  
  190.    
  191.     try:
  192.         #  Takes from input boxes in UI
  193.         dragbool = dragcheck.get()
  194.         angle = float(angleinp.get())
  195.         velocity = float(velocityinp.get().strip()) ############################################
  196.  
  197.         # Gets the gravity and starting height, defaulting to 0 m and 9.81 m respectively if empty
  198.         if not gravityinp.get():
  199.             gravity=9.81
  200.         else:
  201.             gravity=float(gravityinp.get())
  202.  
  203.         if not ystartinp.get():      
  204.             height=0
  205.         else:
  206.             height= float(ystartinp.get())
  207.  
  208.  
  209.         # Valulues are converted to metric if the user has selected imperial units
  210.         if units == "Imperial":
  211.             velocity = velocity / 3.28084
  212.             gravity = gravity / 3.28084
  213.             height= height / 3.28084
  214.  
  215.         if 0 < angle <= 90 and velocity > 0 and gravity > 0 and height >= 0:    # Input validation
  216.             if dragbool == True:
  217.                 mass = float(massinp.get())
  218.                 diameter = float(diameterinp.get())
  219.                 if not dragcoefinp.get():
  220.                     dragcoef = 0.47
  221.                 else:
  222.                     dragcoef = float(dragcoefinp.get())
  223.                
  224.                 if mass > 0 and diameter > 0 and dragcoef > 0:
  225.                     if units == "Imperial":
  226.                         mass = mass / 2.20462
  227.                         diameter = diameter / 3.28084
  228.  
  229.                     xpoints, ypoints, x_at_ymax, ymax, frames, frametime = airres(angle, velocity, gravity, mass, diameter, height, dragcoef)
  230.                     graph(xpoints, ypoints, x_at_ymax, ymax, frames, frametime)
  231.                 else:
  232.                     warning("Mass, diameter and drag coefficient values are required for air resistance.\nThey must all be above 0.")
  233.  
  234.             elif dragbool == False:
  235.                 xpoints, ypoints, x_at_ymax, ymax, frames, frametime = noairres(angle, velocity, gravity, height)
  236.                 graph(xpoints, ypoints, x_at_ymax, ymax, frames, frametime)
  237.         else:
  238.             warning("Angle, velocity, gravity and height values are required.\nAngle must be between 0 and 90 degrees.\nVelocity, gravity and starting height must be above 0.")
  239.     except ValueError or TypeError:
  240.         warning("Do not leave values blank.\nMake sure all values are numbers.")
  241.  
  242. def noairres(theta,u,g,h):
  243.     t=0
  244.     y=0
  245.     xpoints=[]
  246.     ypoints=[]
  247.     frames=[]
  248.     theta=math.radians(theta) # Degrees to radians
  249.  
  250.     #Simplified SUVAT equations are used
  251.     x_at_ymax=u*math.cos(theta)*(u*math.sin(theta))/g   # Calculates the x coordinate of the maximum height
  252.     ymax= x_at_ymax*math.tan(theta)-g*(x_at_ymax**2)*(1+math.tan(theta)**2)/(2*u**2)+h # y coordinate of the maximum height
  253.  
  254.  
  255.     tmax= (u*math.sin(theta)+math.sqrt((u*math.sin(theta))**2+2*g*h))/g  # Calculates the maximum time of flight
  256.     step=tmax/1000    # Keeps the step small enough, but also not too small
  257.  
  258.     while y>=0:
  259.         # SUVAT equations
  260.         x= u*math.cos(theta)*t  
  261.         y= u*math.sin(theta)*t-0.5*g*t**2 + h  # Translates upwards based on the starting height
  262.  
  263.         xpoints.append(x)
  264.         ypoints.append(y)
  265.         frames.append(go.Frame(data=[go.Scatter(x=xpoints, y=ypoints, mode='lines')]))
  266.  
  267.         t+=step
  268.  
  269.    
  270.     # All steps have been completed, the current time is the maximum time
  271.     tmax = t
  272.     frameduration = (tmax * 1000) / len(frames)  # The duration of each frame in milliseconds
  273.    
  274.     # Converts the values back to imperial if the user has selected imperial units
  275.     if units=="Imperial":
  276.         xpoints = [x * 3.28084 for x in xpoints]
  277.         ypoints = [y * 3.28084 for y in ypoints]
  278.  
  279.     return xpoints,ypoints,x_at_ymax,ymax,frames,frameduration
  280.  
  281. def airres(theta,u,g,m,d,h,cd):
  282.     def calculating_Accel(t, state, m):
  283.         x, y, vx, vy = state  # Unnpacks the state list
  284.         v = np.sqrt(vx**2 + vy**2)  # Pythagoream theorem
  285.        
  286.         k = 0.5*cd*1.225*math.pi*(d/2)**2  # Drag coefficient
  287.  
  288.         xF_air = -k * v * vx
  289.         yF_air = -k * v * vy
  290.         ax = xF_air / m
  291.         ay = (yF_air - m * g) / m
  292.        
  293.         return [vx, vy, ax, ay]
  294.    
  295.    
  296.     theta = np.radians(theta)
  297.     vx0 = u * np.cos(theta)
  298.     vy0 = u * np.sin(theta)
  299.  
  300.     # Estimates the maximum time, as there is no analytical solution
  301.     tmax_estimate = (u*math.sin(theta)+math.sqrt((u*math.sin(theta))**2+2*g*h))/g
  302.     step = tmax_estimate / 1000     # Keeps the step small enough, but also not too small
  303.    
  304.     state = np.array([0 , h , vx0 , vy0])  # Initial state of the projectile (x, y, vx, vy)
  305.     t = 0
  306.     ymax = 0
  307.     frames = []
  308.     xpoints = [0]
  309.     ypoints = [h]  # The starting height is set as the first point
  310.    
  311.     while state[1] >= 0:
  312.         # Runge-Kutta 4th order method to find the projectile's path
  313.         k1 = np.array(calculating_Accel(t, state,m)) * step
  314.         k2 = np.array(calculating_Accel(t + step / 2, state + k1 / 2,m)) * step
  315.         k3 = np.array(calculating_Accel(t + step / 2, state + k2 / 2,m)) * step
  316.         k4 = np.array(calculating_Accel(t + step, state + k3,m)) * step
  317.  
  318.         state=state + 1 / 6 * (k1+2*k2+2*k3+k4)
  319.        
  320.         xpoints.append(state[0])  # Appends the x and y coordinates to the list
  321.         ypoints.append(state[1])  
  322.         frames.append(go.Frame(data=[go.Scatter(x=xpoints, y=ypoints, mode='lines')]))
  323.  
  324.         # state[1] is the y coordinate of the projectile, it calculates the coordinates at the maximum
  325.         if state[1] >= ymax:
  326.             ymax=state[1]
  327.             x_at_ymax=state[0]
  328.        
  329.         t += step
  330.  
  331.     # All steps have been completed, the current time is the maximum time
  332.     tmax = t
  333.     frameduration = (tmax * 1000) / len(frames)  # The duration of each frame in milliseconds
  334.  
  335.     return xpoints,ypoints,x_at_ymax,ymax, frames, frameduration
  336.  
  337. def graph(xpoints,ypoints,x_at_ymax,ymax,frames,frametime):
  338.  
  339.     # Checks the current settings set by the user to determine the labels
  340.     if units == "Imperial":
  341.         x_label = "Horizontal Displacement (ft)"
  342.         y_label = "Vertical Displacement (ft)"
  343.     else:
  344.         x_label = "Horizontal Displacement (m)"
  345.         y_label = "Vertical Displacement (m)"
  346.  
  347.     # Creates the graph and animates it
  348.     line = go.Figure(
  349.         data=[go.Scatter(x=[xpoints[0]], y=[ypoints[0]], mode='lines', name='Projectile', line=dict(color='blue'))],
  350.         layout=go.Layout(
  351.             title="Projectile Motion Animation",
  352.             xaxis=dict(title=x_label,zeroline=True, zerolinewidth=2, zerolinecolor='lightblue'),  # Colours the axes to make them more visible
  353.             yaxis=dict(title=y_label,zeroline=True, zerolinewidth=2, zerolinecolor='lightblue'),
  354.             updatemenus=[
  355.                 dict(
  356.                     type="buttons",
  357.                     showactive=True,
  358.                     buttons=[
  359.                         dict(label="Play", method="animate", args=[None, {"frame": {"duration": frametime, "redraw": True}, "fromcurrent": False}]),   # Play and pause buttons
  360.                         dict(label="Pause", method="animate", args=[[None], {"mode": "immediate"}]),],)],),
  361.         frames=frames)
  362.  
  363.     #Creates maximum point marker
  364.     line.add_trace(go.Scatter(x=[x_at_ymax],y=[ymax], mode='markers+text', name='Maximum Point', text=f'Max: ({x_at_ymax:.2f},{ymax:.2f})',textposition='top center', marker=dict(color='blue',size=10,symbol='circle')))
  365.    
  366.     # Creates the graph in a HTML file
  367.     filename = "plot.html"
  368.     line.write_html(filename)
  369.    
  370.     # Opens the html file in a webview window
  371.     webview.create_window("Interactive Graph", filename, width=1920, height=1080, hidden=False)
  372.     webview.start()
  373.  
  374. def settings(event=None):  
  375.     def changetheme(selected):    
  376.         global theme
  377.         theme=selected.lower()
  378.  
  379.         mainwindow.destroy()
  380.         settingwindow.destroy()
  381.         main()
  382.  
  383.     def changebuttons(selected):
  384.         global accentcolour
  385.         if selected=="Blue":
  386.             accentcolour="blue"
  387.         elif selected=="Dark blue":
  388.             accentcolour="dark-blue"
  389.         else:
  390.             accentcolour="green"
  391.  
  392.         mainwindow.destroy()
  393.         settingwindow.destroy()
  394.         main()
  395.        
  396.     def changeunits(selected):
  397.         global units
  398.         units=selected
  399.  
  400.         mainwindow.destroy()
  401.         settingwindow.destroy()
  402.         main()
  403.  
  404.     def changecolour(location, selected):
  405.         global backgroundcolour,framecolour,labelcolour
  406.         if location=="background":
  407.             if selected=="Default":
  408.                 backgroundcolour=['gray92', 'gray14']   # Default customtkinter colour
  409.             else:
  410.                 backgroundcolour=selected
  411.  
  412.         elif location=="frame":
  413.             if selected=="Default":
  414.                 framecolour=['gray86', 'gray17']
  415.             else:
  416.                 framecolour=selected
  417.  
  418.         elif location=="label":
  419.             if selected=="Default":
  420.                 labelcolour=['gray10', '#DCE4EE']
  421.             else:
  422.                 labelcolour=selected
  423.  
  424.         mainwindow.destroy()
  425.         settingwindow.destroy()
  426.         main()
  427.  
  428.     global settingwindow       # Checks if the settings window is already open, and prevents it from opening again if it is
  429.     try:
  430.         if 'normal' == settingwindow.state():
  431.             return
  432.     except:
  433.             pass
  434.    
  435.     # The settings window
  436.     settingwindow=customtkinter.CTk(fg_color=backgroundcolour)
  437.     settingwindow.geometry("300x700")
  438.     settingwindow.resizable(False,False)
  439.     settingwindow.title("Settings")
  440.  
  441.     # The settings frame
  442.     settingsframe=customtkinter.CTkFrame(settingwindow,fg_color=framecolour)
  443.     settingsframe.pack(pady=(20,20),padx=(20,20),fill="both",expand=True)
  444.    
  445.     # Creates the labels and option menu for each setting (theme, units, background color, frame colour, button colour, text colour)
  446.     themetext=customtkinter.CTkLabel(settingsframe, fg_color="transparent", text="Change theme:",text_color=labelcolour)
  447.     themetext.place(x=62,y=20)
  448.     themepick=customtkinter.CTkOptionMenu(settingsframe, values=["Dark","Light"], command=lambda selected: changetheme(selected))
  449.     themepick.set(theme.capitalize())
  450.     themepick.pack(pady=(50,20))
  451.  
  452.     unittext=customtkinter.CTkLabel(settingsframe, fg_color="transparent", text= "Change units:",text_color=labelcolour)
  453.     unittext.place(x=62,y=90)
  454.     unitpick=customtkinter.CTkOptionMenu(settingsframe, values=["Metric", "Imperial"],command=lambda selected: changeunits(selected) )
  455.     unitpick.set(units)
  456.     unitpick.pack(pady=20)
  457.  
  458.     backgroundtext=customtkinter.CTkLabel(settingsframe, fg_color="transparent", text="Change background colour:",text_color=labelcolour)  # Background colour
  459.     backgroundtext.place(x=62,y=158)
  460.     backgroundpick=customtkinter.CTkOptionMenu(settingsframe, values=["Default","Black","White","Snow","Light Slate Gray","Medium Sea Green","Light Goldenrod","Light Sky Blue","Indian Red","Hot Pink","Medium Purple"],command=lambda selected: changecolour("background",selected))
  461.     if backgroundcolour==['gray92', 'gray14']: # Default customtkinter colour
  462.         backgroundpick.set("Default")
  463.     else:
  464.         backgroundpick.set(backgroundcolour)
  465.     backgroundpick.pack(pady=20)
  466.  
  467.     frametext=customtkinter.CTkLabel(settingsframe, fg_color="transparent", text="Change frame colour:",text_color=labelcolour)  # Frame colour
  468.     frametext.place(x=62,y=226)
  469.     framepick=customtkinter.CTkOptionMenu(settingsframe, values=["Default","Black","White","Snow","Light Slate Gray","Medium Sea Green","Light Goldenrod","Light Sky Blue","Indian Red","Hot Pink","Medium Purple"],command=lambda selected: changecolour("frame",selected))
  470.     if framecolour==['gray86', 'gray17']:
  471.         framepick.set("Default")
  472.     else:
  473.         framepick.set(framecolour)
  474.     framepick.pack(pady=20)
  475.  
  476.     global accentcolour
  477.     buttontext=customtkinter.CTkLabel(settingsframe, fg_color="transparent",text="Change buttons colour:",text_color=labelcolour)  # Button colour
  478.     buttontext.place(x=62,y=294)
  479.     buttonpick=customtkinter.CTkOptionMenu(settingsframe, values=["Blue", "Dark blue", "Green"],command=lambda selected: changebuttons(selected))
  480.     buttonpick.set(accentcolour.capitalize())
  481.     buttonpick.pack(pady=20)
  482.  
  483.     labeltext=customtkinter.CTkLabel(settingsframe, fg_color="transparent", text="Change text colour:",text_color=labelcolour)      #Text colour
  484.     labeltext.place(x=62,y=362)
  485.     labelpick=customtkinter.CTkOptionMenu(settingsframe, values=["Default","Black","White"],command=lambda selected: changecolour("label",selected))
  486.     if labelcolour==['gray10', '#DCE4EE']:
  487.         labelpick.set("Default")
  488.     else:
  489.         labelpick.set(labelcolour)
  490.     labelpick.pack(pady=20)
  491.  
  492.  
  493.     mainwindow.protocol("WM_DELETE_WINDOW", lambda: close(settingwindow))  # Closes the settings window and the main window when the 'X' button on the main window is pressed
  494.  
  495.  
  496.     # Keybinds
  497.     settingwindow.bind("<Escape>", lambda e: sys.exit())  # 'Escape' closes the window
  498.  
  499.     settingwindow.mainloop()
  500.  
  501. def close(window):
  502.     try:
  503.         window.destroy()
  504.         mainwindow.destroy()
  505.     except:
  506.         mainwindow.destroy()
  507.  
  508.  
  509.  
  510.  
  511. ## Main code ##
  512.  
  513. # Sets default settings at startup
  514. backgroundcolour=['gray92', 'gray14']
  515. framecolour=['gray86', 'gray17']
  516. labelcolour=['gray10', '#DCE4EE']
  517. theme="system"
  518. accentcolour="blue"
  519. units="Metric"
  520.  
  521.  
  522. main()
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement